mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
1034 lines
39 KiB
Rust
1034 lines
39 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/. */
|
|
|
|
use app_units::{Au, AU_PER_PX};
|
|
use document_loader::{LoadType, LoadBlocker};
|
|
use dom::activation::Activatable;
|
|
use dom::attr::Attr;
|
|
use dom::bindings::cell::DOMRefCell;
|
|
use dom::bindings::codegen::Bindings::DOMRectBinding::DOMRectBinding::DOMRectMethods;
|
|
use dom::bindings::codegen::Bindings::ElementBinding::ElementBinding::ElementMethods;
|
|
use dom::bindings::codegen::Bindings::HTMLImageElementBinding;
|
|
use dom::bindings::codegen::Bindings::HTMLImageElementBinding::HTMLImageElementMethods;
|
|
use dom::bindings::codegen::Bindings::MouseEventBinding::MouseEventMethods;
|
|
use dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
|
|
use dom::bindings::error::Fallible;
|
|
use dom::bindings::inheritance::Castable;
|
|
use dom::bindings::js::{LayoutJS, MutNullableJS, Root};
|
|
use dom::bindings::refcounted::Trusted;
|
|
use dom::bindings::reflector::DomObject;
|
|
use dom::bindings::str::DOMString;
|
|
use dom::document::Document;
|
|
use dom::element::{AttributeMutation, Element, RawLayoutElementHelpers};
|
|
use dom::element::{reflect_cross_origin_attribute, set_cross_origin_attribute};
|
|
use dom::event::{Event, EventBubbles, EventCancelable};
|
|
use dom::eventtarget::EventTarget;
|
|
use dom::htmlareaelement::HTMLAreaElement;
|
|
use dom::htmlelement::HTMLElement;
|
|
use dom::htmlformelement::{FormControl, HTMLFormElement};
|
|
use dom::htmlmapelement::HTMLMapElement;
|
|
use dom::mouseevent::MouseEvent;
|
|
use dom::node::{Node, NodeDamage, document_from_node, window_from_node};
|
|
use dom::progressevent::ProgressEvent;
|
|
use dom::values::UNSIGNED_LONG_MAX;
|
|
use dom::virtualmethods::VirtualMethods;
|
|
use dom::window::Window;
|
|
use dom_struct::dom_struct;
|
|
use euclid::Point2D;
|
|
use html5ever::{LocalName, Prefix};
|
|
use ipc_channel::ipc;
|
|
use ipc_channel::router::ROUTER;
|
|
use microtask::{Microtask, MicrotaskRunnable};
|
|
use net_traits::{FetchResponseListener, FetchMetadata, NetworkError, FetchResponseMsg};
|
|
use net_traits::image::base::{Image, ImageMetadata};
|
|
use net_traits::image_cache::{CanRequestImages, ImageCache, ImageOrMetadataAvailable};
|
|
use net_traits::image_cache::{ImageResponder, ImageResponse, ImageState, PendingImageId};
|
|
use net_traits::image_cache::UsePlaceholder;
|
|
use net_traits::request::{RequestInit, Type as RequestType};
|
|
use network_listener::{NetworkListener, PreInvoke};
|
|
use num_traits::ToPrimitive;
|
|
use script_thread::{Runnable, ScriptThread};
|
|
use servo_url::ServoUrl;
|
|
use servo_url::origin::ImmutableOrigin;
|
|
use std::cell::{Cell, RefMut};
|
|
use std::default::Default;
|
|
use std::i32;
|
|
use std::sync::{Arc, Mutex};
|
|
use style::attr::{AttrValue, LengthOrPercentageOrAuto};
|
|
use task_source::TaskSource;
|
|
|
|
#[derive(Clone, Copy, JSTraceable, HeapSizeOf)]
|
|
#[allow(dead_code)]
|
|
enum State {
|
|
Unavailable,
|
|
PartiallyAvailable,
|
|
CompletelyAvailable,
|
|
Broken,
|
|
}
|
|
#[derive(Copy, Clone, JSTraceable, HeapSizeOf)]
|
|
enum ImageRequestPhase {
|
|
Pending,
|
|
Current
|
|
}
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
#[must_root]
|
|
struct ImageRequest {
|
|
state: State,
|
|
parsed_url: Option<ServoUrl>,
|
|
source_url: Option<DOMString>,
|
|
blocker: Option<LoadBlocker>,
|
|
#[ignore_heap_size_of = "Arc"]
|
|
image: Option<Arc<Image>>,
|
|
metadata: Option<ImageMetadata>,
|
|
final_url: Option<ServoUrl>,
|
|
}
|
|
#[dom_struct]
|
|
pub struct HTMLImageElement {
|
|
htmlelement: HTMLElement,
|
|
image_request: Cell<ImageRequestPhase>,
|
|
current_request: DOMRefCell<ImageRequest>,
|
|
pending_request: DOMRefCell<ImageRequest>,
|
|
form_owner: MutNullableJS<HTMLFormElement>,
|
|
generation: Cell<u32>,
|
|
}
|
|
|
|
impl HTMLImageElement {
|
|
pub fn get_url(&self) -> Option<ServoUrl> {
|
|
self.current_request.borrow().parsed_url.clone()
|
|
}
|
|
}
|
|
|
|
struct ImageResponseHandlerRunnable {
|
|
element: Trusted<HTMLImageElement>,
|
|
image: ImageResponse,
|
|
generation: u32,
|
|
}
|
|
|
|
impl ImageResponseHandlerRunnable {
|
|
fn new(element: Trusted<HTMLImageElement>, image: ImageResponse, generation: u32)
|
|
-> ImageResponseHandlerRunnable {
|
|
ImageResponseHandlerRunnable {
|
|
element: element,
|
|
image: image,
|
|
generation: generation,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Runnable for ImageResponseHandlerRunnable {
|
|
fn name(&self) -> &'static str { "ImageResponseHandlerRunnable" }
|
|
|
|
fn handler(self: Box<Self>) {
|
|
let element = self.element.root();
|
|
// Ignore any image response for a previous request that has been discarded.
|
|
if element.generation.get() == self.generation {
|
|
element.process_image_response(self.image);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// The context required for asynchronously loading an external image.
|
|
struct ImageContext {
|
|
/// Reference to the script thread image cache.
|
|
image_cache: Arc<ImageCache>,
|
|
/// Indicates whether the request failed, and why
|
|
status: Result<(), NetworkError>,
|
|
/// The cache ID for this request.
|
|
id: PendingImageId,
|
|
}
|
|
|
|
impl FetchResponseListener for ImageContext {
|
|
fn process_request_body(&mut self) {}
|
|
fn process_request_eof(&mut self) {}
|
|
|
|
fn process_response(&mut self, metadata: Result<FetchMetadata, NetworkError>) {
|
|
self.image_cache.notify_pending_response(
|
|
self.id,
|
|
FetchResponseMsg::ProcessResponse(metadata.clone()));
|
|
|
|
let metadata = metadata.ok().map(|meta| {
|
|
match meta {
|
|
FetchMetadata::Unfiltered(m) => m,
|
|
FetchMetadata::Filtered { unsafe_, .. } => unsafe_
|
|
}
|
|
});
|
|
|
|
let status_code = metadata.as_ref().and_then(|m| {
|
|
m.status.as_ref().map(|&(code, _)| code)
|
|
}).unwrap_or(0);
|
|
|
|
self.status = match status_code {
|
|
0 => Err(NetworkError::Internal("No http status code received".to_owned())),
|
|
200...299 => Ok(()), // HTTP ok status codes
|
|
_ => Err(NetworkError::Internal(format!("HTTP error code {}", status_code)))
|
|
};
|
|
}
|
|
|
|
fn process_response_chunk(&mut self, payload: Vec<u8>) {
|
|
if self.status.is_ok() {
|
|
self.image_cache.notify_pending_response(
|
|
self.id,
|
|
FetchResponseMsg::ProcessResponseChunk(payload));
|
|
}
|
|
}
|
|
|
|
fn process_response_eof(&mut self, response: Result<(), NetworkError>) {
|
|
self.image_cache.notify_pending_response(
|
|
self.id,
|
|
FetchResponseMsg::ProcessResponseEOF(response));
|
|
}
|
|
}
|
|
|
|
impl PreInvoke for ImageContext {}
|
|
|
|
impl HTMLImageElement {
|
|
/// Update the current image with a valid URL.
|
|
fn fetch_image(&self, img_url: &ServoUrl) {
|
|
fn add_cache_listener_for_element(image_cache: Arc<ImageCache>,
|
|
id: PendingImageId,
|
|
elem: &HTMLImageElement) {
|
|
let trusted_node = Trusted::new(elem);
|
|
let (responder_sender, responder_receiver) = ipc::channel().unwrap();
|
|
|
|
let window = window_from_node(elem);
|
|
let task_source = window.networking_task_source();
|
|
let wrapper = window.get_runnable_wrapper();
|
|
let generation = elem.generation.get();
|
|
ROUTER.add_route(responder_receiver.to_opaque(), box move |message| {
|
|
debug!("Got image {:?}", message);
|
|
// Return the image via a message to the script thread, which marks
|
|
// the element as dirty and triggers a reflow.
|
|
let runnable = ImageResponseHandlerRunnable::new(
|
|
trusted_node.clone(), message.to().unwrap(), generation);
|
|
let _ = task_source.queue_with_wrapper(box runnable, &wrapper);
|
|
});
|
|
|
|
image_cache.add_listener(id, ImageResponder::new(responder_sender, id));
|
|
}
|
|
|
|
let window = window_from_node(self);
|
|
let image_cache = window.image_cache();
|
|
let response =
|
|
image_cache.find_image_or_metadata(img_url.clone().into(),
|
|
UsePlaceholder::Yes,
|
|
CanRequestImages::Yes);
|
|
match response {
|
|
Ok(ImageOrMetadataAvailable::ImageAvailable(image, url)) => {
|
|
self.process_image_response(ImageResponse::Loaded(image, url));
|
|
}
|
|
|
|
Ok(ImageOrMetadataAvailable::MetadataAvailable(m)) => {
|
|
self.process_image_response(ImageResponse::MetadataLoaded(m));
|
|
}
|
|
|
|
Err(ImageState::Pending(id)) => {
|
|
add_cache_listener_for_element(image_cache.clone(), id, self);
|
|
}
|
|
|
|
Err(ImageState::LoadError) => {
|
|
self.process_image_response(ImageResponse::None);
|
|
}
|
|
|
|
Err(ImageState::NotRequested(id)) => {
|
|
add_cache_listener_for_element(image_cache, id, self);
|
|
self.fetch_request(img_url, id);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn fetch_request(&self, img_url: &ServoUrl, id: PendingImageId) {
|
|
let document = document_from_node(self);
|
|
let window = window_from_node(self);
|
|
|
|
let context = Arc::new(Mutex::new(ImageContext {
|
|
image_cache: window.image_cache(),
|
|
status: Ok(()),
|
|
id: id,
|
|
}));
|
|
|
|
let (action_sender, action_receiver) = ipc::channel().unwrap();
|
|
let listener = NetworkListener {
|
|
context: context,
|
|
task_source: window.networking_task_source(),
|
|
wrapper: Some(window.get_runnable_wrapper()),
|
|
};
|
|
ROUTER.add_route(action_receiver.to_opaque(), box move |message| {
|
|
listener.notify_fetch(message.to().unwrap());
|
|
});
|
|
|
|
let request = RequestInit {
|
|
url: img_url.clone(),
|
|
origin: document.origin().immutable().clone(),
|
|
type_: RequestType::Image,
|
|
pipeline_id: Some(document.global().pipeline_id()),
|
|
.. RequestInit::default()
|
|
};
|
|
|
|
// This is a background load because the load blocker already fulfills the
|
|
// purpose of delaying the document's load event.
|
|
document.loader().fetch_async_background(request, action_sender);
|
|
}
|
|
|
|
/// Step 14 of https://html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn process_image_response(&self, image: ImageResponse) {
|
|
// TODO: Handle multipart/x-mixed-replace
|
|
let (trigger_image_load, trigger_image_error) = match (image, self.image_request.get()) {
|
|
(ImageResponse::Loaded(image, url), ImageRequestPhase::Current) |
|
|
(ImageResponse::PlaceholderLoaded(image, url), ImageRequestPhase::Current) => {
|
|
self.current_request.borrow_mut().metadata = Some(ImageMetadata {
|
|
height: image.height,
|
|
width: image.width
|
|
});
|
|
self.current_request.borrow_mut().final_url = Some(url);
|
|
self.current_request.borrow_mut().image = Some(image);
|
|
self.current_request.borrow_mut().state = State::CompletelyAvailable;
|
|
LoadBlocker::terminate(&mut self.current_request.borrow_mut().blocker);
|
|
// Mark the node dirty
|
|
self.upcast::<Node>().dirty(NodeDamage::OtherNodeDamage);
|
|
(true, false)
|
|
},
|
|
(ImageResponse::Loaded(image, url), ImageRequestPhase::Pending) |
|
|
(ImageResponse::PlaceholderLoaded(image, url), ImageRequestPhase::Pending) => {
|
|
self.abort_request(State::Unavailable, ImageRequestPhase::Pending);
|
|
self.image_request.set(ImageRequestPhase::Current);
|
|
self.current_request.borrow_mut().metadata = Some(ImageMetadata {
|
|
height: image.height,
|
|
width: image.width
|
|
});
|
|
self.current_request.borrow_mut().final_url = Some(url);
|
|
self.current_request.borrow_mut().image = Some(image);
|
|
self.current_request.borrow_mut().state = State::CompletelyAvailable;
|
|
LoadBlocker::terminate(&mut self.current_request.borrow_mut().blocker);
|
|
self.upcast::<Node>().dirty(NodeDamage::OtherNodeDamage);
|
|
(true, false)
|
|
},
|
|
(ImageResponse::MetadataLoaded(meta), ImageRequestPhase::Current) => {
|
|
self.current_request.borrow_mut().state = State::PartiallyAvailable;
|
|
self.current_request.borrow_mut().metadata = Some(meta);
|
|
(false, false)
|
|
},
|
|
(ImageResponse::MetadataLoaded(_), ImageRequestPhase::Pending) => {
|
|
self.pending_request.borrow_mut().state = State::PartiallyAvailable;
|
|
(false, false)
|
|
},
|
|
(ImageResponse::None, ImageRequestPhase::Current) => {
|
|
self.abort_request(State::Broken, ImageRequestPhase::Current);
|
|
(false, true)
|
|
},
|
|
(ImageResponse::None, ImageRequestPhase::Pending) => {
|
|
self.abort_request(State::Broken, ImageRequestPhase::Current);
|
|
self.abort_request(State::Broken, ImageRequestPhase::Pending);
|
|
self.image_request.set(ImageRequestPhase::Current);
|
|
(false, true)
|
|
},
|
|
};
|
|
|
|
// Fire image.onload and loadend
|
|
if trigger_image_load {
|
|
// TODO: https://html.spec.whatwg.org/multipage/#fire-a-progress-event-or-event
|
|
self.upcast::<EventTarget>().fire_event(atom!("load"));
|
|
self.upcast::<EventTarget>().fire_event(atom!("loadend"));
|
|
}
|
|
|
|
// Fire image.onerror
|
|
if trigger_image_error {
|
|
self.upcast::<EventTarget>().fire_event(atom!("error"));
|
|
self.upcast::<EventTarget>().fire_event(atom!("loadend"));
|
|
}
|
|
|
|
// Trigger reflow
|
|
let window = window_from_node(self);
|
|
window.add_pending_reflow();
|
|
}
|
|
|
|
/// https://html.spec.whatwg.org/multipage/#abort-the-image-request
|
|
fn abort_request(&self, state: State, request: ImageRequestPhase) {
|
|
let mut request = match request {
|
|
ImageRequestPhase::Current => self.current_request.borrow_mut(),
|
|
ImageRequestPhase::Pending => self.pending_request.borrow_mut(),
|
|
};
|
|
LoadBlocker::terminate(&mut request.blocker);
|
|
request.state = state;
|
|
request.image = None;
|
|
request.metadata = None;
|
|
}
|
|
|
|
/// Step 11.4 of https://html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn set_current_request_url_to_selected_fire_error_and_loadend(&self, src: DOMString) {
|
|
struct Task {
|
|
img: Trusted<HTMLImageElement>,
|
|
src: String,
|
|
}
|
|
impl Runnable for Task {
|
|
fn handler(self: Box<Self>) {
|
|
let img = self.img.root();
|
|
{
|
|
let mut current_request = img.current_request.borrow_mut();
|
|
current_request.source_url = Some(DOMString::from_string(self.src));
|
|
}
|
|
img.upcast::<EventTarget>().fire_event(atom!("error"));
|
|
img.upcast::<EventTarget>().fire_event(atom!("loadend"));
|
|
img.abort_request(State::Broken, ImageRequestPhase::Current);
|
|
img.abort_request(State::Broken, ImageRequestPhase::Pending);
|
|
}
|
|
}
|
|
|
|
let task = box Task {
|
|
img: Trusted::new(self),
|
|
src: src.into()
|
|
};
|
|
let document = document_from_node(self);
|
|
let window = document.window();
|
|
let task_source = window.dom_manipulation_task_source();
|
|
let _ = task_source.queue(task, window.upcast());
|
|
}
|
|
|
|
/// Step 10 of html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn dispatch_loadstart_progress_event(&self) {
|
|
struct FireprogressEventTask {
|
|
img: Trusted<HTMLImageElement>,
|
|
}
|
|
impl Runnable for FireprogressEventTask {
|
|
fn handler(self: Box<Self>) {
|
|
let progressevent = ProgressEvent::new(&self.img.root().global(),
|
|
atom!("loadstart"), EventBubbles::DoesNotBubble, EventCancelable::NotCancelable,
|
|
false, 0, 0);
|
|
progressevent.upcast::<Event>().fire(self.img.root().upcast());
|
|
}
|
|
}
|
|
let runnable = box FireprogressEventTask {
|
|
img: Trusted::new(self),
|
|
};
|
|
let document = document_from_node(self);
|
|
let window = document.window();
|
|
let task = window.dom_manipulation_task_source();
|
|
let _ = task.queue(runnable, window.upcast());
|
|
}
|
|
|
|
/// https://html.spec.whatwg.org/multipage/#update-the-source-set
|
|
fn update_source_set(&self) -> Vec<DOMString> {
|
|
let elem = self.upcast::<Element>();
|
|
// TODO: follow the algorithm
|
|
let src = elem.get_string_attribute(&local_name!("src"));
|
|
if src.is_empty() {
|
|
return vec![]
|
|
}
|
|
vec![src]
|
|
}
|
|
|
|
/// https://html.spec.whatwg.org/multipage/#select-an-image-source
|
|
fn select_image_source(&self) -> Option<DOMString> {
|
|
// TODO: select an image source from source set
|
|
self.update_source_set().first().cloned()
|
|
}
|
|
|
|
/// Step 9.2 of https://html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn set_current_request_url_to_none_fire_error(&self) {
|
|
struct SetUrlToNoneTask {
|
|
img: Trusted<HTMLImageElement>,
|
|
}
|
|
impl Runnable for SetUrlToNoneTask {
|
|
fn handler(self: Box<Self>) {
|
|
let img = self.img.root();
|
|
{
|
|
let mut current_request = img.current_request.borrow_mut();
|
|
current_request.source_url = None;
|
|
current_request.parsed_url = None;
|
|
}
|
|
let elem = img.upcast::<Element>();
|
|
if elem.has_attribute(&local_name!("src")) {
|
|
img.upcast::<EventTarget>().fire_event(atom!("error"));
|
|
}
|
|
img.abort_request(State::Broken, ImageRequestPhase::Current);
|
|
img.abort_request(State::Broken, ImageRequestPhase::Pending);
|
|
}
|
|
}
|
|
|
|
let task = box SetUrlToNoneTask {
|
|
img: Trusted::new(self),
|
|
};
|
|
let document = document_from_node(self);
|
|
let window = document.window();
|
|
let task_source = window.dom_manipulation_task_source();
|
|
let _ = task_source.queue(task, window.upcast());
|
|
}
|
|
|
|
/// Step 5.3.7 of https://html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn set_current_request_url_to_string_and_fire_load(&self, src: DOMString, url: ServoUrl) {
|
|
struct SetUrlToStringTask {
|
|
img: Trusted<HTMLImageElement>,
|
|
src: String,
|
|
url: ServoUrl
|
|
}
|
|
impl Runnable for SetUrlToStringTask {
|
|
fn handler(self: Box<Self>) {
|
|
let img = self.img.root();
|
|
{
|
|
let mut current_request = img.current_request.borrow_mut();
|
|
current_request.parsed_url = Some(self.url.clone());
|
|
current_request.source_url = Some(self.src.into());
|
|
}
|
|
// TODO: restart animation, if set
|
|
img.upcast::<EventTarget>().fire_event(atom!("load"));
|
|
}
|
|
}
|
|
let runnable = box SetUrlToStringTask {
|
|
img: Trusted::new(self),
|
|
src: src.into(),
|
|
url: url
|
|
};
|
|
let document = document_from_node(self);
|
|
let window = document.window();
|
|
let task = window.dom_manipulation_task_source();
|
|
let _ = task.queue(runnable, window.upcast());
|
|
}
|
|
|
|
fn init_image_request(&self,
|
|
request: &mut RefMut<ImageRequest>,
|
|
url: &ServoUrl,
|
|
src: &DOMString) {
|
|
request.parsed_url = Some(url.clone());
|
|
request.source_url = Some(src.clone());
|
|
request.image = None;
|
|
request.metadata = None;
|
|
let document = document_from_node(self);
|
|
LoadBlocker::terminate(&mut request.blocker);
|
|
request.blocker = Some(LoadBlocker::new(&*document, LoadType::Image(url.clone())));
|
|
}
|
|
|
|
/// Step 12 of html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn prepare_image_request(&self, url: &ServoUrl, src: &DOMString) {
|
|
match self.image_request.get() {
|
|
ImageRequestPhase::Pending => {
|
|
if let Some(pending_url) = self.pending_request.borrow().parsed_url.clone() {
|
|
// Step 12.1
|
|
if pending_url == *url {
|
|
return
|
|
}
|
|
}
|
|
},
|
|
ImageRequestPhase::Current => {
|
|
let mut current_request = self.current_request.borrow_mut();
|
|
let mut pending_request = self.pending_request.borrow_mut();
|
|
// step 12.4, create a new "image_request"
|
|
match (current_request.parsed_url.clone(), current_request.state) {
|
|
(Some(parsed_url), State::PartiallyAvailable) => {
|
|
// Step 12.2
|
|
if parsed_url == *url {
|
|
// 12.3 abort pending request
|
|
pending_request.image = None;
|
|
pending_request.parsed_url = None;
|
|
LoadBlocker::terminate(&mut pending_request.blocker);
|
|
// TODO: queue a task to restart animation, if restart-animation is set
|
|
return
|
|
}
|
|
self.image_request.set(ImageRequestPhase::Pending);
|
|
self.init_image_request(&mut pending_request, &url, &src);
|
|
},
|
|
(_, State::Broken) | (_, State::Unavailable) => {
|
|
// Step 12.5
|
|
self.init_image_request(&mut current_request, &url, &src);
|
|
},
|
|
(_, _) => {
|
|
// step 12.6
|
|
self.image_request.set(ImageRequestPhase::Pending);
|
|
self.init_image_request(&mut pending_request, &url, &src);
|
|
},
|
|
}
|
|
}
|
|
}
|
|
self.fetch_image(&url);
|
|
}
|
|
|
|
/// Step 8-12 of html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn update_the_image_data_sync_steps(&self) {
|
|
let document = document_from_node(self);
|
|
// Step 8
|
|
// TODO: take pixel density into account
|
|
match self.select_image_source() {
|
|
Some(src) => {
|
|
// Step 10
|
|
self.dispatch_loadstart_progress_event();
|
|
// Step 11
|
|
let base_url = document.base_url();
|
|
let parsed_url = base_url.join(&src);
|
|
match parsed_url {
|
|
Ok(url) => {
|
|
// Step 12
|
|
self.prepare_image_request(&url, &src);
|
|
},
|
|
Err(_) => {
|
|
// Step 11.1-11.5
|
|
self.set_current_request_url_to_selected_fire_error_and_loadend(src);
|
|
}
|
|
}
|
|
},
|
|
None => {
|
|
// Step 9
|
|
self.set_current_request_url_to_none_fire_error();
|
|
},
|
|
}
|
|
}
|
|
|
|
/// https://html.spec.whatwg.org/multipage/#update-the-image-data
|
|
fn update_the_image_data(&self) {
|
|
let document = document_from_node(self);
|
|
let window = document.window();
|
|
let elem = self.upcast::<Element>();
|
|
let src = elem.get_string_attribute(&local_name!("src"));
|
|
let base_url = document.base_url();
|
|
|
|
// https://html.spec.whatwg.org/multipage/#reacting-to-dom-mutations
|
|
// Always first set the current request to unavailable,
|
|
// ensuring img.complete is false.
|
|
{
|
|
let mut current_request = self.current_request.borrow_mut();
|
|
current_request.state = State::Unavailable;
|
|
}
|
|
|
|
if !document.is_active() {
|
|
// Step 1 (if the document is inactive)
|
|
// TODO: use GlobalScope::enqueue_microtask,
|
|
// to queue micro task to come back to this algorithm
|
|
}
|
|
// Step 2 abort if user-agent does not supports images
|
|
// NOTE: Servo only supports images, skipping this step
|
|
|
|
// step 3, 4
|
|
// TODO: take srcset and parent images into account
|
|
if !src.is_empty() {
|
|
// TODO: take pixel density into account
|
|
if let Ok(img_url) = base_url.join(&src) {
|
|
// step 5, check the list of available images
|
|
let image_cache = window.image_cache();
|
|
let response = image_cache.find_image_or_metadata(img_url.clone().into(),
|
|
UsePlaceholder::No,
|
|
CanRequestImages::No);
|
|
if let Ok(ImageOrMetadataAvailable::ImageAvailable(image, url)) = response {
|
|
// Step 5.3
|
|
let metadata = ImageMetadata { height: image.height, width: image.width };
|
|
// Step 5.3.2 abort requests
|
|
self.abort_request(State::CompletelyAvailable, ImageRequestPhase::Current);
|
|
self.abort_request(State::CompletelyAvailable, ImageRequestPhase::Pending);
|
|
let mut current_request = self.current_request.borrow_mut();
|
|
current_request.final_url = Some(url);
|
|
current_request.image = Some(image.clone());
|
|
current_request.metadata = Some(metadata);
|
|
self.set_current_request_url_to_string_and_fire_load(src, img_url);
|
|
return
|
|
}
|
|
}
|
|
}
|
|
// step 6, await a stable state.
|
|
self.generation.set(self.generation.get() + 1);
|
|
let task = ImageElementMicrotask::StableStateUpdateImageDataTask {
|
|
elem: Root::from_ref(self),
|
|
generation: self.generation.get(),
|
|
};
|
|
ScriptThread::await_stable_state(Microtask::ImageElement(task));
|
|
}
|
|
|
|
fn new_inherited(local_name: LocalName, prefix: Option<Prefix>, document: &Document) -> HTMLImageElement {
|
|
HTMLImageElement {
|
|
htmlelement: HTMLElement::new_inherited(local_name, prefix, document),
|
|
image_request: Cell::new(ImageRequestPhase::Current),
|
|
current_request: DOMRefCell::new(ImageRequest {
|
|
state: State::Unavailable,
|
|
parsed_url: None,
|
|
source_url: None,
|
|
image: None,
|
|
metadata: None,
|
|
blocker: None,
|
|
final_url: None,
|
|
}),
|
|
pending_request: DOMRefCell::new(ImageRequest {
|
|
state: State::Unavailable,
|
|
parsed_url: None,
|
|
source_url: None,
|
|
image: None,
|
|
metadata: None,
|
|
blocker: None,
|
|
final_url: None,
|
|
}),
|
|
form_owner: Default::default(),
|
|
generation: Default::default(),
|
|
}
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
pub fn new(local_name: LocalName,
|
|
prefix: Option<Prefix>,
|
|
document: &Document) -> Root<HTMLImageElement> {
|
|
Node::reflect_node(box HTMLImageElement::new_inherited(local_name, prefix, document),
|
|
document,
|
|
HTMLImageElementBinding::Wrap)
|
|
}
|
|
|
|
pub fn Image(window: &Window,
|
|
width: Option<u32>,
|
|
height: Option<u32>) -> Fallible<Root<HTMLImageElement>> {
|
|
let document = window.Document();
|
|
let image = HTMLImageElement::new(local_name!("img"), None, &document);
|
|
if let Some(w) = width {
|
|
image.SetWidth(w);
|
|
}
|
|
if let Some(h) = height {
|
|
image.SetHeight(h);
|
|
}
|
|
|
|
Ok(image)
|
|
}
|
|
pub fn areas(&self) -> Option<Vec<Root<HTMLAreaElement>>> {
|
|
let elem = self.upcast::<Element>();
|
|
let usemap_attr = match elem.get_attribute(&ns!(), &local_name!("usemap")) {
|
|
Some(attr) => attr,
|
|
None => return None,
|
|
};
|
|
|
|
let value = usemap_attr.value();
|
|
|
|
if value.len() == 0 || !value.is_char_boundary(1) {
|
|
return None
|
|
}
|
|
|
|
let (first, last) = value.split_at(1);
|
|
|
|
if first != "#" || last.len() == 0 {
|
|
return None
|
|
}
|
|
|
|
let useMapElements = document_from_node(self).upcast::<Node>()
|
|
.traverse_preorder()
|
|
.filter_map(Root::downcast::<HTMLMapElement>)
|
|
.find(|n| n.upcast::<Element>().get_string_attribute(&LocalName::from("name")) == last);
|
|
|
|
useMapElements.map(|mapElem| mapElem.get_area_elements())
|
|
}
|
|
|
|
pub fn get_origin(&self) -> Option<ImmutableOrigin> {
|
|
match self.current_request.borrow_mut().final_url {
|
|
Some(ref url) => Some(url.origin()),
|
|
None => None
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
#[derive(JSTraceable, HeapSizeOf)]
|
|
pub enum ImageElementMicrotask {
|
|
StableStateUpdateImageDataTask {
|
|
elem: Root<HTMLImageElement>,
|
|
generation: u32,
|
|
}
|
|
}
|
|
|
|
impl MicrotaskRunnable for ImageElementMicrotask {
|
|
fn handler(&self) {
|
|
match self {
|
|
&ImageElementMicrotask::StableStateUpdateImageDataTask { ref elem, ref generation } => {
|
|
// Step 7 of https://html.spec.whatwg.org/multipage/#update-the-image-data,
|
|
// stop here if other instances of this algorithm have been scheduled
|
|
if elem.generation.get() == *generation {
|
|
elem.update_the_image_data_sync_steps();
|
|
}
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
pub trait LayoutHTMLImageElementHelpers {
|
|
#[allow(unsafe_code)]
|
|
unsafe fn image(&self) -> Option<Arc<Image>>;
|
|
|
|
#[allow(unsafe_code)]
|
|
unsafe fn image_url(&self) -> Option<ServoUrl>;
|
|
|
|
fn get_width(&self) -> LengthOrPercentageOrAuto;
|
|
fn get_height(&self) -> LengthOrPercentageOrAuto;
|
|
}
|
|
|
|
impl LayoutHTMLImageElementHelpers for LayoutJS<HTMLImageElement> {
|
|
#[allow(unsafe_code)]
|
|
unsafe fn image(&self) -> Option<Arc<Image>> {
|
|
(*self.unsafe_get()).current_request.borrow_for_layout().image.clone()
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
unsafe fn image_url(&self) -> Option<ServoUrl> {
|
|
(*self.unsafe_get()).current_request.borrow_for_layout().parsed_url.clone()
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
fn get_width(&self) -> LengthOrPercentageOrAuto {
|
|
unsafe {
|
|
(*self.upcast::<Element>().unsafe_get())
|
|
.get_attr_for_layout(&ns!(), &local_name!("width"))
|
|
.map(AttrValue::as_dimension)
|
|
.cloned()
|
|
.unwrap_or(LengthOrPercentageOrAuto::Auto)
|
|
}
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
fn get_height(&self) -> LengthOrPercentageOrAuto {
|
|
unsafe {
|
|
(*self.upcast::<Element>().unsafe_get())
|
|
.get_attr_for_layout(&ns!(), &local_name!("height"))
|
|
.map(AttrValue::as_dimension)
|
|
.cloned()
|
|
.unwrap_or(LengthOrPercentageOrAuto::Auto)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl HTMLImageElementMethods for HTMLImageElement {
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-alt
|
|
make_getter!(Alt, "alt");
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-alt
|
|
make_setter!(SetAlt, "alt");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-src
|
|
make_url_getter!(Src, "src");
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-src
|
|
make_setter!(SetSrc, "src");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-crossOrigin
|
|
fn GetCrossOrigin(&self) -> Option<DOMString> {
|
|
reflect_cross_origin_attribute(self.upcast::<Element>())
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-crossOrigin
|
|
fn SetCrossOrigin(&self, value: Option<DOMString>) {
|
|
set_cross_origin_attribute(self.upcast::<Element>(), value);
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-usemap
|
|
make_getter!(UseMap, "usemap");
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-usemap
|
|
make_setter!(SetUseMap, "usemap");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-ismap
|
|
make_bool_getter!(IsMap, "ismap");
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-ismap
|
|
make_bool_setter!(SetIsMap, "ismap");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-width
|
|
fn Width(&self) -> u32 {
|
|
let node = self.upcast::<Node>();
|
|
match node.bounding_content_box() {
|
|
Some(rect) => rect.size.width.to_px() as u32,
|
|
None => self.NaturalWidth(),
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-width
|
|
fn SetWidth(&self, value: u32) {
|
|
image_dimension_setter(self.upcast(), local_name!("width"), value);
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-height
|
|
fn Height(&self) -> u32 {
|
|
let node = self.upcast::<Node>();
|
|
match node.bounding_content_box() {
|
|
Some(rect) => rect.size.height.to_px() as u32,
|
|
None => self.NaturalHeight(),
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-height
|
|
fn SetHeight(&self, value: u32) {
|
|
image_dimension_setter(self.upcast(), local_name!("height"), value);
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-naturalwidth
|
|
fn NaturalWidth(&self) -> u32 {
|
|
let ref metadata = self.current_request.borrow().metadata;
|
|
|
|
match *metadata {
|
|
Some(ref metadata) => metadata.width,
|
|
None => 0,
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-naturalheight
|
|
fn NaturalHeight(&self) -> u32 {
|
|
let ref metadata = self.current_request.borrow().metadata;
|
|
|
|
match *metadata {
|
|
Some(ref metadata) => metadata.height,
|
|
None => 0,
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-complete
|
|
fn Complete(&self) -> bool {
|
|
let elem = self.upcast::<Element>();
|
|
// TODO: take srcset into account
|
|
if !elem.has_attribute(&local_name!("src")) {
|
|
return true
|
|
}
|
|
let src = elem.get_string_attribute(&local_name!("src"));
|
|
if src.is_empty() {
|
|
return true
|
|
}
|
|
let request = self.current_request.borrow();
|
|
let request_state = request.state;
|
|
match request_state {
|
|
State::CompletelyAvailable | State::Broken => return true,
|
|
State::PartiallyAvailable | State::Unavailable => return false,
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-currentsrc
|
|
fn CurrentSrc(&self) -> DOMString {
|
|
let ref url = self.current_request.borrow().source_url;
|
|
match *url {
|
|
Some(ref url) => url.clone(),
|
|
None => DOMString::from(""),
|
|
}
|
|
}
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-name
|
|
make_getter!(Name, "name");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-name
|
|
make_atomic_setter!(SetName, "name");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-align
|
|
make_getter!(Align, "align");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-align
|
|
make_setter!(SetAlign, "align");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-hspace
|
|
make_uint_getter!(Hspace, "hspace");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-hspace
|
|
make_uint_setter!(SetHspace, "hspace");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-vspace
|
|
make_uint_getter!(Vspace, "vspace");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-vspace
|
|
make_uint_setter!(SetVspace, "vspace");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-longdesc
|
|
make_getter!(LongDesc, "longdesc");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-longdesc
|
|
make_setter!(SetLongDesc, "longdesc");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-border
|
|
make_getter!(Border, "border");
|
|
|
|
// https://html.spec.whatwg.org/multipage/#dom-img-border
|
|
make_setter!(SetBorder, "border");
|
|
}
|
|
|
|
impl VirtualMethods for HTMLImageElement {
|
|
fn super_type(&self) -> Option<&VirtualMethods> {
|
|
Some(self.upcast::<HTMLElement>() as &VirtualMethods)
|
|
}
|
|
|
|
fn adopting_steps(&self, old_doc: &Document) {
|
|
self.super_type().unwrap().adopting_steps(old_doc);
|
|
self.update_the_image_data();
|
|
}
|
|
|
|
fn attribute_mutated(&self, attr: &Attr, mutation: AttributeMutation) {
|
|
self.super_type().unwrap().attribute_mutated(attr, mutation);
|
|
match attr.local_name() {
|
|
&local_name!("src") => self.update_the_image_data(),
|
|
_ => {},
|
|
}
|
|
}
|
|
|
|
fn parse_plain_attribute(&self, name: &LocalName, value: DOMString) -> AttrValue {
|
|
match name {
|
|
&local_name!("name") => AttrValue::from_atomic(value.into()),
|
|
&local_name!("width") | &local_name!("height") => AttrValue::from_dimension(value.into()),
|
|
&local_name!("hspace") | &local_name!("vspace") => AttrValue::from_u32(value.into(), 0),
|
|
_ => self.super_type().unwrap().parse_plain_attribute(name, value),
|
|
}
|
|
}
|
|
|
|
fn handle_event(&self, event: &Event) {
|
|
if event.type_() != atom!("click") {
|
|
return
|
|
}
|
|
|
|
let area_elements = self.areas();
|
|
let elements = match area_elements {
|
|
Some(x) => x,
|
|
None => return,
|
|
};
|
|
|
|
// Fetch click coordinates
|
|
let mouse_event = match event.downcast::<MouseEvent>() {
|
|
Some(x) => x,
|
|
None => return,
|
|
};
|
|
|
|
let point = Point2D::new(mouse_event.ClientX().to_f32().unwrap(),
|
|
mouse_event.ClientY().to_f32().unwrap());
|
|
let bcr = self.upcast::<Element>().GetBoundingClientRect();
|
|
let bcr_p = Point2D::new(bcr.X() as f32, bcr.Y() as f32);
|
|
|
|
// Walk HTMLAreaElements
|
|
for element in elements {
|
|
let shape = element.get_shape_from_coords();
|
|
let shp = match shape {
|
|
Some(x) => x.absolute_coords(bcr_p),
|
|
None => return,
|
|
};
|
|
if shp.hit_test(&point) {
|
|
element.activation_behavior(event, self.upcast());
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl FormControl for HTMLImageElement {
|
|
fn form_owner(&self) -> Option<Root<HTMLFormElement>> {
|
|
self.form_owner.get()
|
|
}
|
|
|
|
fn set_form_owner(&self, form: Option<&HTMLFormElement>) {
|
|
self.form_owner.set(form);
|
|
}
|
|
|
|
fn to_element<'a>(&'a self) -> &'a Element {
|
|
self.upcast::<Element>()
|
|
}
|
|
|
|
fn is_listed(&self) -> bool {
|
|
false
|
|
}
|
|
}
|
|
|
|
fn image_dimension_setter(element: &Element, attr: LocalName, value: u32) {
|
|
// This setter is a bit weird: the IDL type is unsigned long, but it's parsed as
|
|
// a dimension for rendering.
|
|
let value = if value > UNSIGNED_LONG_MAX {
|
|
0
|
|
} else {
|
|
value
|
|
};
|
|
|
|
// FIXME: There are probably quite a few more cases of this. This is the
|
|
// only overflow that was hitting on automation, but we should consider what
|
|
// to do in the general case case.
|
|
//
|
|
// See <https://github.com/servo/app_units/issues/22>
|
|
let pixel_value = if value > (i32::MAX / AU_PER_PX) as u32 {
|
|
0
|
|
} else {
|
|
value
|
|
};
|
|
|
|
let dim = LengthOrPercentageOrAuto::Length(Au::from_px(pixel_value as i32));
|
|
let value = AttrValue::Dimension(value.to_string(), dim);
|
|
element.set_attribute(&attr, value);
|
|
}
|