diff options
author | Maharsh <maharsh312@gmail.com> | 2019-05-14 02:15:20 -0400 |
---|---|---|
committer | Josh Matthews <josh@joshmatthews.net> | 2019-05-22 10:24:54 -0400 |
commit | 85c20db495b25af653e6cb77130e166fae8d4b20 (patch) | |
tree | c19296c2d3108b6ca59a4ecb8ca3e12cf2a0389b /components/script/dom/canvasrenderingcontext2d.rs | |
parent | 6fb7a8cdc787abf7c69304d3186b0a318ef25412 (diff) | |
download | servo-85c20db495b25af653e6cb77130e166fae8d4b20.tar.gz servo-85c20db495b25af653e6cb77130e166fae8d4b20.zip |
Extract canvas operations for reuse by OffscreenCanvas.
Diffstat (limited to 'components/script/dom/canvasrenderingcontext2d.rs')
-rw-r--r-- | components/script/dom/canvasrenderingcontext2d.rs | 1950 |
1 files changed, 1217 insertions, 733 deletions
diff --git a/components/script/dom/canvasrenderingcontext2d.rs b/components/script/dom/canvasrenderingcontext2d.rs index 0639e70549a..2a26a1c5752 100644 --- a/components/script/dom/canvasrenderingcontext2d.rs +++ b/components/script/dom/canvasrenderingcontext2d.rs @@ -23,7 +23,7 @@ use crate::dom::element::Element; use crate::dom::globalscope::GlobalScope; use crate::dom::htmlcanvaselement::{CanvasContext, HTMLCanvasElement}; use crate::dom::imagedata::ImageData; -use crate::dom::node::{window_from_node, Node, NodeDamage}; +use crate::dom::node::{Node, NodeDamage}; use crate::unpremultiplytable::UNPREMULTIPLY_TABLE; use canvas_traits::canvas::{Canvas2dMsg, CanvasId, CanvasMsg}; use canvas_traits::canvas::{CompositionOrBlending, FillOrStrokeStyle, FillRule}; @@ -65,16 +65,6 @@ pub struct CanvasRenderingContext2D { /// For rendering contexts created by an HTML canvas element, this is Some, /// for ones created by a paint worklet, this is None. canvas: Option<Dom<HTMLCanvasElement>>, - #[ignore_malloc_size_of = "Arc"] - image_cache: Arc<ImageCache>, - /// Any missing image URLs. - missing_image_urls: DomRefCell<Vec<ServoUrl>>, - /// The base URL for resolving CSS image URL values. - /// Needed because of https://github.com/servo/servo/issues/17625 - base_url: ServoUrl, - state: DomRefCell<CanvasContextState>, - saved_states: DomRefCell<Vec<CanvasContextState>>, - origin_clean: Cell<bool>, canvas_state: DomRefCell<CanvasState>, } @@ -119,11 +109,22 @@ impl CanvasContextState { } } +#[must_root] #[derive(JSTraceable, MallocSizeOf)] pub struct CanvasState { #[ignore_malloc_size_of = "Defined in ipc-channel"] ipc_renderer: IpcSender<CanvasMsg>, canvas_id: CanvasId, + state: DomRefCell<CanvasContextState>, + origin_clean: Cell<bool>, + #[ignore_malloc_size_of = "Arc"] + image_cache: Arc<dyn ImageCache>, + /// The base URL for resolving CSS image URL values. + /// Needed because of https://github.com/servo/servo/issues/17625 + base_url: ServoUrl, + /// Any missing image URLs. + missing_image_urls: DomRefCell<Vec<ServoUrl>>, + saved_states: DomRefCell<Vec<CanvasContextState>>, } impl CanvasState { @@ -141,6 +142,12 @@ impl CanvasState { CanvasState { ipc_renderer: ipc_renderer, canvas_id: canvas_id, + state: DomRefCell::new(CanvasContextState::new()), + origin_clean: Cell::new(true), + image_cache: global.image_cache(), + base_url: global.api_base_url(), + missing_image_urls: DomRefCell::new(Vec::new()), + saved_states: DomRefCell::new(Vec::new()), } } @@ -169,172 +176,122 @@ impl CanvasState { )) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-fillrect - pub fn FillRect(&self, x: f64, y: f64, width: f64, height: f64) { - if let Some(rect) = self.create_drawable_rect(x, y, width, height) { - self.send_canvas_2d_msg(Canvas2dMsg::FillRect(rect)); - } - } - - // https://html.spec.whatwg.org/multipage/#dom-context-2d-clearrect - pub fn ClearRect(&self, x: f64, y: f64, width: f64, height: f64) { - if let Some(rect) = self.create_drawable_rect(x, y, width, height) { - self.send_canvas_2d_msg(Canvas2dMsg::ClearRect(rect)); - } + fn origin_is_clean(&self) -> bool { + self.origin_clean.get() } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokerect - pub fn StrokeRect(&self, x: f64, y: f64, width: f64, height: f64) { - if let Some(rect) = self.create_drawable_rect(x, y, width, height) { - self.send_canvas_2d_msg(Canvas2dMsg::StrokeRect(rect)); - } + fn set_origin_unclean(&self) { + self.origin_clean.set(false) } -} -impl CanvasRenderingContext2D { - pub fn new_inherited( - global: &GlobalScope, - canvas: Option<&HTMLCanvasElement>, - image_cache: Arc<dyn ImageCache>, - base_url: ServoUrl, - size: Size2D<u32>, - ) -> CanvasRenderingContext2D { - CanvasRenderingContext2D { - reflector_: Reflector::new(), - canvas: canvas.map(Dom::from_ref), - image_cache: image_cache, - missing_image_urls: DomRefCell::new(Vec::new()), - base_url: base_url, - state: DomRefCell::new(CanvasContextState::new()), - saved_states: DomRefCell::new(Vec::new()), - origin_clean: Cell::new(true), - canvas_state: DomRefCell::new(CanvasState::new( - global, - Size2D::new(size.width as u64, size.height as u64), - )), + // https://html.spec.whatwg.org/multipage/#the-image-argument-is-not-origin-clean + fn is_origin_clean(&self, image: CanvasImageSource) -> bool { + match image { + CanvasImageSource::HTMLCanvasElement(canvas) => canvas.origin_is_clean(), + CanvasImageSource::HTMLImageElement(image) => { + image.same_origin(GlobalScope::entry().origin()) + }, + CanvasImageSource::CSSStyleValue(_) => true, } } - pub fn new( - global: &GlobalScope, - canvas: &HTMLCanvasElement, - size: Size2D<u32>, - ) -> DomRoot<CanvasRenderingContext2D> { - let window = window_from_node(canvas); - let image_cache = window.image_cache(); - let base_url = window.get_url(); - let boxed = Box::new(CanvasRenderingContext2D::new_inherited( - global, - Some(canvas), - image_cache, - base_url, - size, - )); - reflect_dom_object(boxed, global, CanvasRenderingContext2DBinding::Wrap) - } + fn fetch_image_data(&self, url: ServoUrl) -> Option<(Vec<u8>, Size2D<u32>)> { + let img = match self.request_image_from_cache(url) { + ImageResponse::Loaded(img, _) => img, + ImageResponse::PlaceholderLoaded(_, _) | + ImageResponse::None | + ImageResponse::MetadataLoaded(_) => { + return None; + }, + }; - // https://html.spec.whatwg.org/multipage/#concept-canvas-set-bitmap-dimensions - pub fn set_bitmap_dimensions(&self, size: Size2D<u32>) { - self.reset_to_initial_state(); - self.canvas_state - .borrow() - .ipc_renderer - .send(CanvasMsg::Recreate( - size, - self.canvas_state.borrow().get_canvas_id(), - )) - .unwrap(); - } + let image_size = Size2D::new(img.width, img.height); + let image_data = match img.format { + PixelFormat::BGRA8 => img.bytes.to_vec(), + pixel_format => unimplemented!("unsupported pixel format ({:?})", pixel_format), + }; - // https://html.spec.whatwg.org/multipage/#reset-the-rendering-context-to-its-default-state - fn reset_to_initial_state(&self) { - self.saved_states.borrow_mut().clear(); - *self.state.borrow_mut() = CanvasContextState::new(); + Some((image_data, image_size)) } - fn mark_as_dirty(&self) { - if let Some(ref canvas) = self.canvas { - canvas.upcast::<Node>().dirty(NodeDamage::OtherNodeDamage); + #[inline] + fn request_image_from_cache(&self, url: ServoUrl) -> ImageResponse { + let response = self.image_cache.find_image_or_metadata( + url.clone(), + UsePlaceholder::No, + CanRequestImages::No, + ); + match response { + Ok(ImageOrMetadataAvailable::ImageAvailable(image, url)) => { + ImageResponse::Loaded(image, url) + }, + Err(ImageState::Pending(_)) => ImageResponse::None, + _ => { + // Rather annoyingly, we get the same response back from + // A load which really failed and from a load which hasn't started yet. + self.missing_image_urls.borrow_mut().push(url); + ImageResponse::None + }, } } - fn update_transform(&self) { - self.send_canvas_2d_msg(Canvas2dMsg::SetTransform(self.state.borrow().transform)) - } - - // It is used by DrawImage to calculate the size of the source and destination rectangles based - // on the drawImage call arguments - // source rectangle = area of the original image to be copied - // destination rectangle = area of the destination canvas where the source image is going to be drawn - fn adjust_source_dest_rects( - &self, - image_size: Size2D<f64>, - sx: f64, - sy: f64, - sw: f64, - sh: f64, - dx: f64, - dy: f64, - dw: f64, - dh: f64, - ) -> (Rect<f64>, Rect<f64>) { - let image_rect = Rect::new( - Point2D::new(0f64, 0f64), - Size2D::new(image_size.width as f64, image_size.height as f64), - ); - - // The source rectangle is the rectangle whose corners are the four points (sx, sy), - // (sx+sw, sy), (sx+sw, sy+sh), (sx, sy+sh). - let source_rect = Rect::new( - Point2D::new(sx.min(sx + sw), sy.min(sy + sh)), - Size2D::new(sw.abs(), sh.abs()), - ); + fn parse_color(&self, canvas: Option<&HTMLCanvasElement>, string: &str) -> Result<RGBA, ()> { + let mut input = ParserInput::new(string); + let mut parser = Parser::new(&mut input); + let color = CSSColor::parse(&mut parser); + if parser.is_exhausted() { + match color { + Ok(CSSColor::RGBA(rgba)) => Ok(rgba), + Ok(CSSColor::CurrentColor) => { + // TODO: https://github.com/whatwg/html/issues/1099 + // Reconsider how to calculate currentColor in a display:none canvas - // When the source rectangle is outside the source image, - // the source rectangle must be clipped to the source image - let source_rect_clipped = source_rect - .intersection(&image_rect) - .unwrap_or(Rect::zero()); + // TODO: will need to check that the context bitmap mode is fixed + // once we implement CanvasProxy + let canvas = match canvas { + // https://drafts.css-houdini.org/css-paint-api/#2d-rendering-context + // Whenever "currentColor" is used as a color in the PaintRenderingContext2D API, + // it is treated as opaque black. + None => return Ok(RGBA::new(0, 0, 0, 255)), + Some(ref canvas) => &**canvas, + }; - // Width and height ratios between the non clipped and clipped source rectangles - let width_ratio: f64 = source_rect_clipped.size.width / source_rect.size.width; - let height_ratio: f64 = source_rect_clipped.size.height / source_rect.size.height; + let canvas_element = canvas.upcast::<Element>(); - // When the source rectangle is outside the source image, - // the destination rectangle must be clipped in the same proportion. - let dest_rect_width_scaled: f64 = dw * width_ratio; - let dest_rect_height_scaled: f64 = dh * height_ratio; + match canvas_element.style() { + Some(ref s) if canvas_element.has_css_layout_box() => { + Ok(s.get_color().color) + }, + _ => Ok(RGBA::new(0, 0, 0, 255)), + } + }, + _ => Err(()), + } + } else { + Err(()) + } + } - // The destination rectangle is the rectangle whose corners are the four points (dx, dy), - // (dx+dw, dy), (dx+dw, dy+dh), (dx, dy+dh). - let dest_rect = Rect::new( - Point2D::new( - dx.min(dx + dest_rect_width_scaled), - dy.min(dy + dest_rect_height_scaled), - ), - Size2D::new(dest_rect_width_scaled.abs(), dest_rect_height_scaled.abs()), - ); + pub fn get_rect(&self, canvas: Option<&HTMLCanvasElement>, rect: Rect<u32>) -> Vec<u8> { + assert!(self.origin_is_clean()); - let source_rect = Rect::new( - Point2D::new(source_rect_clipped.origin.x, source_rect_clipped.origin.y), - Size2D::new( - source_rect_clipped.size.width, - source_rect_clipped.size.height, - ), - ); + // FIXME(nox): This is probably wrong when this is a context for an + // offscreen canvas. + let canvas_size = canvas.as_ref().map_or(Size2D::zero(), |c| c.get_size()); + assert!(Rect::from_size(canvas_size).contains_rect(&rect)); - (source_rect, dest_rect) - } + let (sender, receiver) = ipc::bytes_channel().unwrap(); + self.send_canvas_2d_msg(Canvas2dMsg::GetImageData(rect, canvas_size, sender)); + let mut pixels = receiver.recv().unwrap().to_vec(); - // https://html.spec.whatwg.org/multipage/#the-image-argument-is-not-origin-clean - fn is_origin_clean(&self, image: CanvasImageSource) -> bool { - match image { - CanvasImageSource::HTMLCanvasElement(canvas) => canvas.origin_is_clean(), - CanvasImageSource::HTMLImageElement(image) => { - image.same_origin(GlobalScope::entry().origin()) - }, - CanvasImageSource::CSSStyleValue(_) => true, + for chunk in pixels.chunks_mut(4) { + let b = chunk[0]; + chunk[0] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + chunk[2] as usize]; + chunk[1] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + chunk[1] as usize]; + chunk[2] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + b as usize]; } + + pixels } // @@ -360,6 +317,7 @@ impl CanvasRenderingContext2D { // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage fn draw_image( &self, + htmlcanvas: Option<&HTMLCanvasElement>, image: CanvasImageSource, sx: f64, sy: f64, @@ -372,20 +330,20 @@ impl CanvasRenderingContext2D { ) -> ErrorResult { let result = match image { CanvasImageSource::HTMLCanvasElement(ref canvas) => { - self.draw_html_canvas_element(&canvas, sx, sy, sw, sh, dx, dy, dw, dh) + self.draw_html_canvas_element(&canvas, htmlcanvas, sx, sy, sw, sh, dx, dy, dw, dh) }, CanvasImageSource::HTMLImageElement(ref image) => { // https://html.spec.whatwg.org/multipage/#img-error // If the image argument is an HTMLImageElement object that is in the broken state, // then throw an InvalidStateError exception let url = image.get_url().ok_or(Error::InvalidState)?; - self.fetch_and_draw_image_data(url, sx, sy, sw, sh, dx, dy, dw, dh) + self.fetch_and_draw_image_data(htmlcanvas, url, sx, sy, sw, sh, dx, dy, dw, dh) }, CanvasImageSource::CSSStyleValue(ref value) => { let url = value .get_url(self.base_url.clone()) .ok_or(Error::InvalidState)?; - self.fetch_and_draw_image_data(url, sx, sy, sw, sh, dx, dy, dw, dh) + self.fetch_and_draw_image_data(htmlcanvas, url, sx, sy, sw, sh, dx, dy, dw, dh) }, }; @@ -398,6 +356,7 @@ impl CanvasRenderingContext2D { fn draw_html_canvas_element( &self, canvas: &HTMLCanvasElement, + htmlcanvas: Option<&HTMLCanvasElement>, sx: f64, sy: f64, sw: Option<f64>, @@ -433,7 +392,7 @@ impl CanvasRenderingContext2D { match *context { CanvasContext::Context2d(ref context) => { context.send_canvas_2d_msg(Canvas2dMsg::DrawImageInOther( - self.canvas_state.borrow().get_canvas_id(), + self.get_canvas_id(), image_size, dest_rect, source_rect, @@ -452,12 +411,13 @@ impl CanvasRenderingContext2D { )); } - self.mark_as_dirty(); + self.mark_as_dirty(htmlcanvas); Ok(()) } fn fetch_and_draw_image_data( &self, + canvas: Option<&HTMLCanvasElement>, url: ServoUrl, sx: f64, sy: f64, @@ -494,182 +454,359 @@ impl CanvasRenderingContext2D { source_rect, smoothing_enabled, )); - self.mark_as_dirty(); + self.mark_as_dirty(canvas); Ok(()) } - fn fetch_image_data(&self, url: ServoUrl) -> Option<(Vec<u8>, Size2D<u32>)> { - let img = match self.request_image_from_cache(url) { - ImageResponse::Loaded(img, _) => img, - ImageResponse::PlaceholderLoaded(_, _) | - ImageResponse::None | - ImageResponse::MetadataLoaded(_) => { - return None; - }, - }; + fn mark_as_dirty(&self, canvas: Option<&HTMLCanvasElement>) { + if let Some(ref canvas) = canvas { + canvas.upcast::<Node>().dirty(NodeDamage::OtherNodeDamage); + } + } - let image_size = Size2D::new(img.width, img.height); - let image_data = match img.format { - PixelFormat::BGRA8 => img.bytes.to_vec(), - pixel_format => unimplemented!("unsupported pixel format ({:?})", pixel_format), - }; + // It is used by DrawImage to calculate the size of the source and destination rectangles based + // on the drawImage call arguments + // source rectangle = area of the original image to be copied + // destination rectangle = area of the destination canvas where the source image is going to be drawn + fn adjust_source_dest_rects( + &self, + image_size: Size2D<f64>, + sx: f64, + sy: f64, + sw: f64, + sh: f64, + dx: f64, + dy: f64, + dw: f64, + dh: f64, + ) -> (Rect<f64>, Rect<f64>) { + let image_rect = Rect::new( + Point2D::new(0f64, 0f64), + Size2D::new(image_size.width as f64, image_size.height as f64), + ); - Some((image_data, image_size)) - } + // The source rectangle is the rectangle whose corners are the four points (sx, sy), + // (sx+sw, sy), (sx+sw, sy+sh), (sx, sy+sh). + let source_rect = Rect::new( + Point2D::new(sx.min(sx + sw), sy.min(sy + sh)), + Size2D::new(sw.abs(), sh.abs()), + ); - #[inline] - fn request_image_from_cache(&self, url: ServoUrl) -> ImageResponse { - let response = self.image_cache.find_image_or_metadata( - url.clone(), - UsePlaceholder::No, - CanRequestImages::No, + // When the source rectangle is outside the source image, + // the source rectangle must be clipped to the source image + let source_rect_clipped = source_rect + .intersection(&image_rect) + .unwrap_or(Rect::zero()); + + // Width and height ratios between the non clipped and clipped source rectangles + let width_ratio: f64 = source_rect_clipped.size.width / source_rect.size.width; + let height_ratio: f64 = source_rect_clipped.size.height / source_rect.size.height; + + // When the source rectangle is outside the source image, + // the destination rectangle must be clipped in the same proportion. + let dest_rect_width_scaled: f64 = dw * width_ratio; + let dest_rect_height_scaled: f64 = dh * height_ratio; + + // The destination rectangle is the rectangle whose corners are the four points (dx, dy), + // (dx+dw, dy), (dx+dw, dy+dh), (dx, dy+dh). + let dest_rect = Rect::new( + Point2D::new( + dx.min(dx + dest_rect_width_scaled), + dy.min(dy + dest_rect_height_scaled), + ), + Size2D::new(dest_rect_width_scaled.abs(), dest_rect_height_scaled.abs()), ); - match response { - Ok(ImageOrMetadataAvailable::ImageAvailable(image, url)) => { - ImageResponse::Loaded(image, url) - }, - Err(ImageState::Pending(_)) => ImageResponse::None, - _ => { - // Rather annoyingly, we get the same response back from - // A load which really failed and from a load which hasn't started yet. - self.missing_image_urls.borrow_mut().push(url); - ImageResponse::None - }, - } - } - pub fn take_missing_image_urls(&self) -> Vec<ServoUrl> { - mem::replace(&mut self.missing_image_urls.borrow_mut(), vec![]) + let source_rect = Rect::new( + Point2D::new(source_rect_clipped.origin.x, source_rect_clipped.origin.y), + Size2D::new( + source_rect_clipped.size.width, + source_rect_clipped.size.height, + ), + ); + + (source_rect, dest_rect) } - fn parse_color(&self, string: &str) -> Result<RGBA, ()> { - let mut input = ParserInput::new(string); - let mut parser = Parser::new(&mut input); - let color = CSSColor::parse(&mut parser); - if parser.is_exhausted() { - match color { - Ok(CSSColor::RGBA(rgba)) => Ok(rgba), - Ok(CSSColor::CurrentColor) => { - // TODO: https://github.com/whatwg/html/issues/1099 - // Reconsider how to calculate currentColor in a display:none canvas + fn update_transform(&self) { + self.send_canvas_2d_msg(Canvas2dMsg::SetTransform(self.state.borrow().transform)) + } - // TODO: will need to check that the context bitmap mode is fixed - // once we implement CanvasProxy - let canvas = match self.canvas { - // https://drafts.css-houdini.org/css-paint-api/#2d-rendering-context - // Whenever "currentColor" is used as a color in the PaintRenderingContext2D API, - // it is treated as opaque black. - None => return Ok(RGBA::new(0, 0, 0, 255)), - Some(ref canvas) => &**canvas, - }; + // https://html.spec.whatwg.org/multipage/#dom-context-2d-fillrect + pub fn FillRect(&self, x: f64, y: f64, width: f64, height: f64) { + if let Some(rect) = self.create_drawable_rect(x, y, width, height) { + self.send_canvas_2d_msg(Canvas2dMsg::FillRect(rect)); + } + } - let canvas_element = canvas.upcast::<Element>(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-clearrect + pub fn ClearRect(&self, x: f64, y: f64, width: f64, height: f64) { + if let Some(rect) = self.create_drawable_rect(x, y, width, height) { + self.send_canvas_2d_msg(Canvas2dMsg::ClearRect(rect)); + } + } - match canvas_element.style() { - Some(ref s) if canvas_element.has_css_layout_box() => { - Ok(s.get_color().color) - }, - _ => Ok(RGBA::new(0, 0, 0, 255)), - } - }, - _ => Err(()), - } - } else { - Err(()) + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokerect + pub fn StrokeRect(&self, x: f64, y: f64, width: f64, height: f64) { + if let Some(rect) = self.create_drawable_rect(x, y, width, height) { + self.send_canvas_2d_msg(Canvas2dMsg::StrokeRect(rect)); } } - pub fn get_canvas_id(&self) -> CanvasId { - self.canvas_state.borrow().get_canvas_id() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsetx + pub fn ShadowOffsetX(&self) -> f64 { + self.state.borrow().shadow_offset_x } - pub fn send_canvas_2d_msg(&self, msg: Canvas2dMsg) { - self.canvas_state.borrow().send_canvas_2d_msg(msg) + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsetx + pub fn SetShadowOffsetX(&self, value: f64) { + if !value.is_finite() || value == self.state.borrow().shadow_offset_x { + return; + } + self.state.borrow_mut().shadow_offset_x = value; + self.send_canvas_2d_msg(Canvas2dMsg::SetShadowOffsetX(value)) } - pub fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg> { - self.canvas_state.borrow().ipc_renderer.clone() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsety + pub fn ShadowOffsetY(&self) -> f64 { + self.state.borrow().shadow_offset_y } - pub fn origin_is_clean(&self) -> bool { - self.origin_clean.get() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsety + pub fn SetShadowOffsetY(&self, value: f64) { + if !value.is_finite() || value == self.state.borrow().shadow_offset_y { + return; + } + self.state.borrow_mut().shadow_offset_y = value; + self.send_canvas_2d_msg(Canvas2dMsg::SetShadowOffsetY(value)) } - fn set_origin_unclean(&self) { - self.origin_clean.set(false) + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowblur + pub fn ShadowBlur(&self) -> f64 { + self.state.borrow().shadow_blur } - pub fn get_rect(&self, rect: Rect<u32>) -> Vec<u8> { - assert!(self.origin_is_clean()); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowblur + pub fn SetShadowBlur(&self, value: f64) { + if !value.is_finite() || value < 0f64 || value == self.state.borrow().shadow_blur { + return; + } + self.state.borrow_mut().shadow_blur = value; + self.send_canvas_2d_msg(Canvas2dMsg::SetShadowBlur(value)) + } - // FIXME(nox): This is probably wrong when this is a context for an - // offscreen canvas. - let canvas_size = self - .canvas - .as_ref() - .map_or(Size2D::zero(), |c| c.get_size()); - assert!(Rect::from_size(canvas_size).contains_rect(&rect)); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowcolor + pub fn ShadowColor(&self) -> DOMString { + let mut result = String::new(); + serialize(&self.state.borrow().shadow_color, &mut result).unwrap(); + DOMString::from(result) + } - let (sender, receiver) = ipc::bytes_channel().unwrap(); - self.send_canvas_2d_msg(Canvas2dMsg::GetImageData(rect, canvas_size, sender)); - let mut pixels = receiver.recv().unwrap().to_vec(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowcolor + pub fn SetShadowColor(&self, value: DOMString) { + if let Ok(color) = parse_color(&value) { + self.state.borrow_mut().shadow_color = color; + self.send_canvas_2d_msg(Canvas2dMsg::SetShadowColor(color)) + } + } - for chunk in pixels.chunks_mut(4) { - let b = chunk[0]; - chunk[0] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + chunk[2] as usize]; - chunk[1] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + chunk[1] as usize]; - chunk[2] = UNPREMULTIPLY_TABLE[256 * (chunk[3] as usize) + b as usize]; + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle + pub fn StrokeStyle(&self) -> StringOrCanvasGradientOrCanvasPattern { + match self.state.borrow().stroke_style { + CanvasFillOrStrokeStyle::Color(ref rgba) => { + let mut result = String::new(); + serialize(rgba, &mut result).unwrap(); + StringOrCanvasGradientOrCanvasPattern::String(DOMString::from(result)) + }, + CanvasFillOrStrokeStyle::Gradient(ref gradient) => { + StringOrCanvasGradientOrCanvasPattern::CanvasGradient(DomRoot::from_ref(&*gradient)) + }, + CanvasFillOrStrokeStyle::Pattern(ref pattern) => { + StringOrCanvasGradientOrCanvasPattern::CanvasPattern(DomRoot::from_ref(&*pattern)) + }, } + } - pixels + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle + pub fn SetStrokeStyle( + &self, + canvas: Option<&HTMLCanvasElement>, + value: StringOrCanvasGradientOrCanvasPattern, + ) { + match value { + StringOrCanvasGradientOrCanvasPattern::String(string) => { + if let Ok(rgba) = self.parse_color(canvas, &string) { + self.state.borrow_mut().stroke_style = CanvasFillOrStrokeStyle::Color(rgba); + self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle(FillOrStrokeStyle::Color( + rgba, + ))); + } + }, + StringOrCanvasGradientOrCanvasPattern::CanvasGradient(gradient) => { + self.state.borrow_mut().stroke_style = + CanvasFillOrStrokeStyle::Gradient(Dom::from_ref(&*gradient)); + self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle( + gradient.to_fill_or_stroke_style(), + )); + }, + StringOrCanvasGradientOrCanvasPattern::CanvasPattern(pattern) => { + self.state.borrow_mut().stroke_style = + CanvasFillOrStrokeStyle::Pattern(Dom::from_ref(&*pattern)); + self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle( + pattern.to_fill_or_stroke_style(), + )); + if !pattern.origin_is_clean() { + self.set_origin_unclean(); + } + }, + } } -} -pub trait LayoutCanvasRenderingContext2DHelpers { - #[allow(unsafe_code)] - unsafe fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg>; - #[allow(unsafe_code)] - unsafe fn get_canvas_id(&self) -> CanvasId; -} + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle + pub fn FillStyle(&self) -> StringOrCanvasGradientOrCanvasPattern { + match self.state.borrow().fill_style { + CanvasFillOrStrokeStyle::Color(ref rgba) => { + let mut result = String::new(); + serialize(rgba, &mut result).unwrap(); + StringOrCanvasGradientOrCanvasPattern::String(DOMString::from(result)) + }, + CanvasFillOrStrokeStyle::Gradient(ref gradient) => { + StringOrCanvasGradientOrCanvasPattern::CanvasGradient(DomRoot::from_ref(&*gradient)) + }, + CanvasFillOrStrokeStyle::Pattern(ref pattern) => { + StringOrCanvasGradientOrCanvasPattern::CanvasPattern(DomRoot::from_ref(&*pattern)) + }, + } + } -impl LayoutCanvasRenderingContext2DHelpers for LayoutDom<CanvasRenderingContext2D> { - #[allow(unsafe_code)] - unsafe fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg> { - (*self.unsafe_get()) - .canvas_state - .borrow_for_layout() - .ipc_renderer - .clone() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle + pub fn SetFillStyle( + &self, + canvas: Option<&HTMLCanvasElement>, + value: StringOrCanvasGradientOrCanvasPattern, + ) { + match value { + StringOrCanvasGradientOrCanvasPattern::String(string) => { + if let Ok(rgba) = self.parse_color(canvas, &string) { + self.state.borrow_mut().fill_style = CanvasFillOrStrokeStyle::Color(rgba); + self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle(FillOrStrokeStyle::Color( + rgba, + ))) + } + }, + StringOrCanvasGradientOrCanvasPattern::CanvasGradient(gradient) => { + self.state.borrow_mut().fill_style = + CanvasFillOrStrokeStyle::Gradient(Dom::from_ref(&*gradient)); + self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle( + gradient.to_fill_or_stroke_style(), + )); + }, + StringOrCanvasGradientOrCanvasPattern::CanvasPattern(pattern) => { + self.state.borrow_mut().fill_style = + CanvasFillOrStrokeStyle::Pattern(Dom::from_ref(&*pattern)); + self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle( + pattern.to_fill_or_stroke_style(), + )); + if !pattern.origin_is_clean() { + self.set_origin_unclean(); + } + }, + } } - #[allow(unsafe_code)] - unsafe fn get_canvas_id(&self) -> CanvasId { - (*self.unsafe_get()) - .canvas_state - .borrow_for_layout() - .get_canvas_id() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-createlineargradient + pub fn CreateLinearGradient( + &self, + global: &GlobalScope, + x0: Finite<f64>, + y0: Finite<f64>, + x1: Finite<f64>, + y1: Finite<f64>, + ) -> DomRoot<CanvasGradient> { + CanvasGradient::new( + global, + CanvasGradientStyle::Linear(LinearGradientStyle::new(*x0, *y0, *x1, *y1, Vec::new())), + ) } -} -// We add a guard to each of methods by the spec: -// http://www.w3.org/html/wg/drafts/2dcontext/html5_canvas_CR/ -// -// > Except where otherwise specified, for the 2D context interface, -// > any method call with a numeric argument whose value is infinite or a NaN value must be ignored. -// -// Restricted values are guarded in glue code. Therefore we need not add a guard. -// -// FIXME: this behavior should might be generated by some annotattions to idl. -impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { - // https://html.spec.whatwg.org/multipage/#dom-context-2d-canvas - fn Canvas(&self) -> DomRoot<HTMLCanvasElement> { - // This method is not called from a paint worklet rendering context, - // so it's OK to panic if self.canvas is None. - DomRoot::from_ref(self.canvas.as_ref().expect("No canvas.")) + // https://html.spec.whatwg.org/multipage/#dom-context-2d-createradialgradient + pub fn CreateRadialGradient( + &self, + global: &GlobalScope, + x0: Finite<f64>, + y0: Finite<f64>, + r0: Finite<f64>, + x1: Finite<f64>, + y1: Finite<f64>, + r1: Finite<f64>, + ) -> Fallible<DomRoot<CanvasGradient>> { + if *r0 < 0. || *r1 < 0. { + return Err(Error::IndexSize); + } + + Ok(CanvasGradient::new( + global, + CanvasGradientStyle::Radial(RadialGradientStyle::new( + *x0, + *y0, + *r0, + *x1, + *y1, + *r1, + Vec::new(), + )), + )) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-createpattern + pub fn CreatePattern( + &self, + global: &GlobalScope, + image: CanvasImageSource, + mut repetition: DOMString, + ) -> Fallible<DomRoot<CanvasPattern>> { + let (image_data, image_size) = match image { + CanvasImageSource::HTMLImageElement(ref image) => { + // https://html.spec.whatwg.org/multipage/#img-error + // If the image argument is an HTMLImageElement object that is in the broken state, + // then throw an InvalidStateError exception + image + .get_url() + .and_then(|url| self.fetch_image_data(url)) + .ok_or(Error::InvalidState)? + }, + CanvasImageSource::HTMLCanvasElement(ref canvas) => { + let (data, size) = canvas.fetch_all_data().ok_or(Error::InvalidState)?; + let data = data + .map(|data| data.to_vec()) + .unwrap_or_else(|| vec![0; size.area() as usize * 4]); + (data, size) + }, + CanvasImageSource::CSSStyleValue(ref value) => value + .get_url(self.base_url.clone()) + .and_then(|url| self.fetch_image_data(url)) + .ok_or(Error::InvalidState)?, + }; + + if repetition.is_empty() { + repetition.push_str("repeat"); + } + + if let Ok(rep) = RepetitionStyle::from_str(&repetition) { + Ok(CanvasPattern::new( + global, + image_data, + image_size, + rep, + self.is_origin_clean(image), + )) + } else { + Err(Error::Syntax) + } } // https://html.spec.whatwg.org/multipage/#dom-context-2d-save - fn Save(&self) { + pub fn Save(&self) { self.saved_states .borrow_mut() .push(self.state.borrow().clone()); @@ -678,7 +815,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { #[allow(unrooted_must_root)] // https://html.spec.whatwg.org/multipage/#dom-context-2d-restore - fn Restore(&self) { + pub fn Restore(&self) { let mut saved_states = self.saved_states.borrow_mut(); if let Some(state) = saved_states.pop() { self.state.borrow_mut().clone_from(&state); @@ -686,200 +823,283 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-scale - fn Scale(&self, x: f64, y: f64) { - if !(x.is_finite() && y.is_finite()) { - return; - } - - let transform = self.state.borrow().transform; - self.state.borrow_mut().transform = transform.pre_scale(x as f32, y as f32); - self.update_transform() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha + pub fn GlobalAlpha(&self) -> f64 { + self.state.borrow().global_alpha } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-rotate - fn Rotate(&self, angle: f64) { - if angle == 0.0 || !angle.is_finite() { + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha + pub fn SetGlobalAlpha(&self, alpha: f64) { + if !alpha.is_finite() || alpha > 1.0 || alpha < 0.0 { return; } - let (sin, cos) = (angle.sin(), angle.cos()); - let transform = self.state.borrow().transform; - self.state.borrow_mut().transform = transform.pre_mul(&Transform2D::row_major( - cos as f32, - sin as f32, - -sin as f32, - cos as f32, - 0.0, - 0.0, - )); - self.update_transform() + self.state.borrow_mut().global_alpha = alpha; + self.send_canvas_2d_msg(Canvas2dMsg::SetGlobalAlpha(alpha as f32)) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-translate - fn Translate(&self, x: f64, y: f64) { - if !(x.is_finite() && y.is_finite()) { - return; + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation + pub fn GlobalCompositeOperation(&self) -> DOMString { + match self.state.borrow().global_composition { + CompositionOrBlending::Composition(op) => DOMString::from(op.to_str()), + CompositionOrBlending::Blending(op) => DOMString::from(op.to_str()), } - - let transform = self.state.borrow().transform; - self.state.borrow_mut().transform = transform.pre_translate(vec2(x as f32, y as f32)); - self.update_transform() } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-transform - fn Transform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { - if !(a.is_finite() && - b.is_finite() && - c.is_finite() && - d.is_finite() && - e.is_finite() && - f.is_finite()) - { - return; + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation + pub fn SetGlobalCompositeOperation(&self, op_str: DOMString) { + if let Ok(op) = CompositionOrBlending::from_str(&op_str) { + self.state.borrow_mut().global_composition = op; + self.send_canvas_2d_msg(Canvas2dMsg::SetGlobalComposition(op)) } - - let transform = self.state.borrow().transform; - self.state.borrow_mut().transform = transform.pre_mul(&Transform2D::row_major( - a as f32, b as f32, c as f32, d as f32, e as f32, f as f32, - )); - self.update_transform() } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-settransform - fn SetTransform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { - if !(a.is_finite() && - b.is_finite() && - c.is_finite() && - d.is_finite() && - e.is_finite() && - f.is_finite()) - { - return; - } + // https://html.spec.whatwg.org/multipage/#dom-context-2d-imagesmoothingenabled + pub fn ImageSmoothingEnabled(&self) -> bool { + self.state.borrow().image_smoothing_enabled + } - self.state.borrow_mut().transform = - Transform2D::row_major(a as f32, b as f32, c as f32, d as f32, e as f32, f as f32); - self.update_transform() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-imagesmoothingenabled + pub fn SetImageSmoothingEnabled(&self, value: bool) { + self.state.borrow_mut().image_smoothing_enabled = value; } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-resettransform - fn ResetTransform(&self) { - self.state.borrow_mut().transform = Transform2D::identity(); - self.update_transform() + // https://html.spec.whatwg.org/multipage/#dom-context-2d-filltext + pub fn FillText(&self, text: DOMString, x: f64, y: f64, max_width: Option<f64>) { + let parsed_text: String = text.into(); + self.send_canvas_2d_msg(Canvas2dMsg::FillText(parsed_text, x, y, max_width)); } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha - fn GlobalAlpha(&self) -> f64 { - let state = self.state.borrow(); - state.global_alpha + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linewidth + pub fn LineWidth(&self) -> f64 { + self.state.borrow().line_width } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha - fn SetGlobalAlpha(&self, alpha: f64) { - if !alpha.is_finite() || alpha > 1.0 || alpha < 0.0 { + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linewidth + pub fn SetLineWidth(&self, width: f64) { + if !width.is_finite() || width <= 0.0 { return; } - self.state.borrow_mut().global_alpha = alpha; - self.send_canvas_2d_msg(Canvas2dMsg::SetGlobalAlpha(alpha as f32)) + self.state.borrow_mut().line_width = width; + self.send_canvas_2d_msg(Canvas2dMsg::SetLineWidth(width as f32)) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation - fn GlobalCompositeOperation(&self) -> DOMString { - let state = self.state.borrow(); - match state.global_composition { - CompositionOrBlending::Composition(op) => DOMString::from(op.to_str()), - CompositionOrBlending::Blending(op) => DOMString::from(op.to_str()), + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linecap + pub fn LineCap(&self) -> CanvasLineCap { + match self.state.borrow().line_cap { + LineCapStyle::Butt => CanvasLineCap::Butt, + LineCapStyle::Round => CanvasLineCap::Round, + LineCapStyle::Square => CanvasLineCap::Square, } } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation - fn SetGlobalCompositeOperation(&self, op_str: DOMString) { - if let Ok(op) = CompositionOrBlending::from_str(&op_str) { - self.state.borrow_mut().global_composition = op; - self.send_canvas_2d_msg(Canvas2dMsg::SetGlobalComposition(op)) - } + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linecap + pub fn SetLineCap(&self, cap: CanvasLineCap) { + let line_cap = match cap { + CanvasLineCap::Butt => LineCapStyle::Butt, + CanvasLineCap::Round => LineCapStyle::Round, + CanvasLineCap::Square => LineCapStyle::Square, + }; + self.state.borrow_mut().line_cap = line_cap; + self.send_canvas_2d_msg(Canvas2dMsg::SetLineCap(line_cap)); } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-fillrect - fn FillRect(&self, x: f64, y: f64, width: f64, height: f64) { - self.canvas_state.borrow().FillRect(x, y, width, height); - self.mark_as_dirty(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linejoin + pub fn LineJoin(&self) -> CanvasLineJoin { + match self.state.borrow().line_join { + LineJoinStyle::Round => CanvasLineJoin::Round, + LineJoinStyle::Bevel => CanvasLineJoin::Bevel, + LineJoinStyle::Miter => CanvasLineJoin::Miter, + } } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-clearrect - fn ClearRect(&self, x: f64, y: f64, width: f64, height: f64) { - self.canvas_state.borrow().ClearRect(x, y, width, height); - self.mark_as_dirty(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-linejoin + pub fn SetLineJoin(&self, join: CanvasLineJoin) { + let line_join = match join { + CanvasLineJoin::Round => LineJoinStyle::Round, + CanvasLineJoin::Bevel => LineJoinStyle::Bevel, + CanvasLineJoin::Miter => LineJoinStyle::Miter, + }; + self.state.borrow_mut().line_join = line_join; + self.send_canvas_2d_msg(Canvas2dMsg::SetLineJoin(line_join)); } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokerect - fn StrokeRect(&self, x: f64, y: f64, width: f64, height: f64) { - self.canvas_state.borrow().StrokeRect(x, y, width, height); - self.mark_as_dirty(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-miterlimit + pub fn MiterLimit(&self) -> f64 { + self.state.borrow().miter_limit } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-beginpath - fn BeginPath(&self) { - self.send_canvas_2d_msg(Canvas2dMsg::BeginPath); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-miterlimit + pub fn SetMiterLimit(&self, limit: f64) { + if !limit.is_finite() || limit <= 0.0 { + return; + } + + self.state.borrow_mut().miter_limit = limit; + self.send_canvas_2d_msg(Canvas2dMsg::SetMiterLimit(limit as f32)) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-closepath - fn ClosePath(&self) { - self.send_canvas_2d_msg(Canvas2dMsg::ClosePath); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-createimagedata + pub fn CreateImageData( + &self, + global: &GlobalScope, + sw: i32, + sh: i32, + ) -> Fallible<DomRoot<ImageData>> { + if sw == 0 || sh == 0 { + return Err(Error::IndexSize); + } + ImageData::new(global, sw.abs() as u32, sh.abs() as u32, None) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-fill - fn Fill(&self, _: CanvasFillRule) { - // TODO: Process fill rule - self.send_canvas_2d_msg(Canvas2dMsg::Fill); - self.mark_as_dirty(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-createimagedata + pub fn CreateImageData_( + &self, + global: &GlobalScope, + imagedata: &ImageData, + ) -> Fallible<DomRoot<ImageData>> { + ImageData::new(global, imagedata.Width(), imagedata.Height(), None) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-stroke - fn Stroke(&self) { - self.send_canvas_2d_msg(Canvas2dMsg::Stroke); - self.mark_as_dirty(); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-getimagedata + pub fn GetImageData( + &self, + canvas: Option<&HTMLCanvasElement>, + global: &GlobalScope, + sx: i32, + sy: i32, + sw: i32, + sh: i32, + ) -> Fallible<DomRoot<ImageData>> { + // FIXME(nox): There are many arithmetic operations here that can + // overflow or underflow, this should probably be audited. + + if sw == 0 || sh == 0 { + return Err(Error::IndexSize); + } + + if !self.origin_is_clean() { + return Err(Error::Security); + } + + let (origin, size) = adjust_size_sign(Point2D::new(sx, sy), Size2D::new(sw, sh)); + // FIXME(nox): This is probably wrong when this is a context for an + // offscreen canvas. + let canvas_size = canvas.as_ref().map_or(Size2D::zero(), |c| c.get_size()); + let read_rect = match pixels::clip(origin, size, canvas_size) { + Some(rect) => rect, + None => { + // All the pixels are outside the canvas surface. + return ImageData::new(global, size.width, size.height, None); + }, + }; + + ImageData::new( + global, + size.width, + size.height, + Some(self.get_rect(canvas, read_rect)), + ) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-clip - fn Clip(&self, _: CanvasFillRule) { - // TODO: Process fill rule - self.send_canvas_2d_msg(Canvas2dMsg::Clip); + // https://html.spec.whatwg.org/multipage/#dom-context-2d-putimagedata + pub fn PutImageData( + &self, + canvas: Option<&HTMLCanvasElement>, + imagedata: &ImageData, + dx: i32, + dy: i32, + ) { + self.PutImageData_( + canvas, + imagedata, + dx, + dy, + 0, + 0, + imagedata.Width() as i32, + imagedata.Height() as i32, + ) } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-ispointinpath - fn IsPointInPath(&self, x: f64, y: f64, fill_rule: CanvasFillRule) -> bool { - let fill_rule = match fill_rule { - CanvasFillRule::Nonzero => FillRule::Nonzero, - CanvasFillRule::Evenodd => FillRule::Evenodd, + // https://html.spec.whatwg.org/multipage/#dom-context-2d-putimagedata + #[allow(unsafe_code)] + pub fn PutImageData_( + &self, + canvas: Option<&HTMLCanvasElement>, + imagedata: &ImageData, + dx: i32, + dy: i32, + dirty_x: i32, + dirty_y: i32, + dirty_width: i32, + dirty_height: i32, + ) { + // FIXME(nox): There are many arithmetic operations here that can + // overflow or underflow, this should probably be audited. + + let imagedata_size = Size2D::new(imagedata.Width(), imagedata.Height()); + if imagedata_size.area() == 0 { + return; + } + + // Step 1. + // Done later. + + // Step 2. + // TODO: throw InvalidState if buffer is detached. + + // FIXME(nox): This is probably wrong when this is a context for an + // offscreen canvas. + let canvas_size = canvas.as_ref().map_or(Size2D::zero(), |c| c.get_size()); + + // Steps 3-6. + let (src_origin, src_size) = adjust_size_sign( + Point2D::new(dirty_x, dirty_y), + Size2D::new(dirty_width, dirty_height), + ); + let src_rect = match pixels::clip(src_origin, src_size, imagedata_size) { + Some(rect) => rect, + None => return, + }; + let (dst_origin, _) = adjust_size_sign( + Point2D::new(dirty_x.saturating_add(dx), dirty_y.saturating_add(dy)), + Size2D::new(dirty_width, dirty_height), + ); + // By clipping to the canvas surface, we avoid sending any pixel + // that would fall outside it. + let dst_rect = match pixels::clip(dst_origin, src_rect.size, canvas_size) { + Some(rect) => rect, + None => return, }; - let (sender, receiver) = - profiled_ipc::channel::<bool>(self.global().time_profiler_chan().clone()).unwrap(); - self.send_canvas_2d_msg(Canvas2dMsg::IsPointInPath(x, y, fill_rule, sender)); - receiver.recv().unwrap() - } - // https://html.spec.whatwg.org/multipage/#dom-context-2d-filltext - fn FillText(&self, text: DOMString, x: f64, y: f64, max_width: Option<f64>) { - let parsed_text: String = text.into(); - self.send_canvas_2d_msg(Canvas2dMsg::FillText(parsed_text, x, y, max_width)); - self.mark_as_dirty(); + // Step 7. + let (sender, receiver) = ipc::bytes_channel().unwrap(); + let pixels = unsafe { &imagedata.get_rect(Rect::new(src_rect.origin, dst_rect.size)) }; + self.send_canvas_2d_msg(Canvas2dMsg::PutImageData(dst_rect, receiver)); + sender.send(pixels).unwrap(); } // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage - fn DrawImage(&self, image: CanvasImageSource, dx: f64, dy: f64) -> ErrorResult { + pub fn DrawImage( + &self, + canvas: Option<&HTMLCanvasElement>, + image: CanvasImageSource, + dx: f64, + dy: f64, + ) -> ErrorResult { if !(dx.is_finite() && dy.is_finite()) { return Ok(()); } - self.draw_image(image, 0f64, 0f64, None, None, dx, dy, None, None) + self.draw_image(canvas, image, 0f64, 0f64, None, None, dx, dy, None, None) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage - fn DrawImage_( + pub fn DrawImage_( &self, + canvas: Option<&HTMLCanvasElement>, image: CanvasImageSource, dx: f64, dy: f64, @@ -890,12 +1110,24 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { return Ok(()); } - self.draw_image(image, 0f64, 0f64, None, None, dx, dy, Some(dw), Some(dh)) + self.draw_image( + canvas, + image, + 0f64, + 0f64, + None, + None, + dx, + dy, + Some(dw), + Some(dh), + ) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage - fn DrawImage__( + pub fn DrawImage__( &self, + canvas: Option<&HTMLCanvasElement>, image: CanvasImageSource, sx: f64, sy: f64, @@ -919,6 +1151,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } self.draw_image( + canvas, image, sx, sy, @@ -931,8 +1164,136 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { ) } + // https://html.spec.whatwg.org/multipage/#dom-context-2d-beginpath + pub fn BeginPath(&self) { + self.send_canvas_2d_msg(Canvas2dMsg::BeginPath); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-fill + pub fn Fill(&self, _fill_rule: CanvasFillRule) { + // TODO: Process fill rule + self.send_canvas_2d_msg(Canvas2dMsg::Fill); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-stroke + pub fn Stroke(&self) { + self.send_canvas_2d_msg(Canvas2dMsg::Stroke); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-clip + pub fn Clip(&self, _fill_rule: CanvasFillRule) { + // TODO: Process fill rule + self.send_canvas_2d_msg(Canvas2dMsg::Clip); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-ispointinpath + pub fn IsPointInPath( + &self, + global: &GlobalScope, + x: f64, + y: f64, + fill_rule: CanvasFillRule, + ) -> bool { + let fill_rule = match fill_rule { + CanvasFillRule::Nonzero => FillRule::Nonzero, + CanvasFillRule::Evenodd => FillRule::Evenodd, + }; + let (sender, receiver) = + profiled_ipc::channel::<bool>(global.time_profiler_chan().clone()).unwrap(); + self.send_canvas_2d_msg(Canvas2dMsg::IsPointInPath(x, y, fill_rule, sender)); + receiver.recv().unwrap() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-scale + pub fn Scale(&self, x: f64, y: f64) { + if !(x.is_finite() && y.is_finite()) { + return; + } + + let transform = self.state.borrow().transform; + self.state.borrow_mut().transform = transform.pre_scale(x as f32, y as f32); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-rotate + pub fn Rotate(&self, angle: f64) { + if angle == 0.0 || !angle.is_finite() { + return; + } + + let (sin, cos) = (angle.sin(), angle.cos()); + let transform = self.state.borrow().transform; + self.state.borrow_mut().transform = transform.pre_mul(&Transform2D::row_major( + cos as f32, + sin as f32, + -sin as f32, + cos as f32, + 0.0, + 0.0, + )); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-translate + pub fn Translate(&self, x: f64, y: f64) { + if !(x.is_finite() && y.is_finite()) { + return; + } + + let transform = self.state.borrow().transform; + self.state.borrow_mut().transform = transform.pre_translate(vec2(x as f32, y as f32)); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-transform + pub fn Transform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { + if !(a.is_finite() && + b.is_finite() && + c.is_finite() && + d.is_finite() && + e.is_finite() && + f.is_finite()) + { + return; + } + + let transform = self.state.borrow().transform; + self.state.borrow_mut().transform = transform.pre_mul(&Transform2D::row_major( + a as f32, b as f32, c as f32, d as f32, e as f32, f as f32, + )); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-settransform + pub fn SetTransform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { + if !(a.is_finite() && + b.is_finite() && + c.is_finite() && + d.is_finite() && + e.is_finite() && + f.is_finite()) + { + return; + } + + self.state.borrow_mut().transform = + Transform2D::row_major(a as f32, b as f32, c as f32, d as f32, e as f32, f as f32); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-resettransform + pub fn ResetTransform(&self) { + self.state.borrow_mut().transform = Transform2D::identity(); + self.update_transform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-closepath + pub fn ClosePath(&self) { + self.send_canvas_2d_msg(Canvas2dMsg::ClosePath); + } + // https://html.spec.whatwg.org/multipage/#dom-context-2d-moveto - fn MoveTo(&self, x: f64, y: f64) { + pub fn MoveTo(&self, x: f64, y: f64) { if !(x.is_finite() && y.is_finite()) { return; } @@ -940,7 +1301,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-lineto - fn LineTo(&self, x: f64, y: f64) { + pub fn LineTo(&self, x: f64, y: f64) { if !(x.is_finite() && y.is_finite()) { return; } @@ -948,7 +1309,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-rect - fn Rect(&self, x: f64, y: f64, width: f64, height: f64) { + pub fn Rect(&self, x: f64, y: f64, width: f64, height: f64) { if [x, y, width, height].iter().all(|val| val.is_finite()) { let rect = Rect::new( Point2D::new(x as f32, y as f32), @@ -959,7 +1320,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-quadraticcurveto - fn QuadraticCurveTo(&self, cpx: f64, cpy: f64, x: f64, y: f64) { + pub fn QuadraticCurveTo(&self, cpx: f64, cpy: f64, x: f64, y: f64) { if !(cpx.is_finite() && cpy.is_finite() && x.is_finite() && y.is_finite()) { return; } @@ -970,7 +1331,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-beziercurveto - fn BezierCurveTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, x: f64, y: f64) { + pub fn BezierCurveTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, x: f64, y: f64) { if !(cp1x.is_finite() && cp1y.is_finite() && cp2x.is_finite() && @@ -988,7 +1349,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-arc - fn Arc(&self, x: f64, y: f64, r: f64, start: f64, end: f64, ccw: bool) -> ErrorResult { + pub fn Arc(&self, x: f64, y: f64, r: f64, start: f64, end: f64, ccw: bool) -> ErrorResult { if !([x, y, r, start, end].iter().all(|x| x.is_finite())) { return Ok(()); } @@ -1008,7 +1369,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-arcto - fn ArcTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, r: f64) -> ErrorResult { + pub fn ArcTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, r: f64) -> ErrorResult { if !([cp1x, cp1y, cp2x, cp2y, r].iter().all(|x| x.is_finite())) { return Ok(()); } @@ -1025,7 +1386,7 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { } // https://html.spec.whatwg.org/multipage/#dom-context-2d-ellipse - fn Ellipse( + pub fn Ellipse( &self, x: f64, y: f64, @@ -1057,173 +1418,430 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { )); Ok(()) } +} + +impl CanvasRenderingContext2D { + pub fn new_inherited( + global: &GlobalScope, + canvas: Option<&HTMLCanvasElement>, + size: Size2D<u32>, + ) -> CanvasRenderingContext2D { + CanvasRenderingContext2D { + reflector_: Reflector::new(), + canvas: canvas.map(Dom::from_ref), + canvas_state: DomRefCell::new(CanvasState::new( + global, + Size2D::new(size.width as u64, size.height as u64), + )), + } + } + + pub fn new( + global: &GlobalScope, + canvas: &HTMLCanvasElement, + size: Size2D<u32>, + ) -> DomRoot<CanvasRenderingContext2D> { + let boxed = Box::new(CanvasRenderingContext2D::new_inherited( + global, + Some(canvas), + size, + )); + reflect_dom_object(boxed, global, CanvasRenderingContext2DBinding::Wrap) + } + + // https://html.spec.whatwg.org/multipage/#concept-canvas-set-bitmap-dimensions + pub fn set_bitmap_dimensions(&self, size: Size2D<u32>) { + self.reset_to_initial_state(); + self.canvas_state + .borrow() + .ipc_renderer + .send(CanvasMsg::Recreate( + size, + self.canvas_state.borrow().get_canvas_id(), + )) + .unwrap(); + } + + // https://html.spec.whatwg.org/multipage/#reset-the-rendering-context-to-its-default-state + fn reset_to_initial_state(&self) { + self.canvas_state.borrow().saved_states.borrow_mut().clear(); + *self.canvas_state.borrow().state.borrow_mut() = CanvasContextState::new(); + } + + fn mark_as_dirty(&self) { + self.canvas_state + .borrow() + .mark_as_dirty(self.canvas.as_ref().map(|c| &**c)) + } + + pub fn take_missing_image_urls(&self) -> Vec<ServoUrl> { + mem::replace( + &mut self.canvas_state.borrow().missing_image_urls.borrow_mut(), + vec![], + ) + } + + pub fn get_canvas_id(&self) -> CanvasId { + self.canvas_state.borrow().get_canvas_id() + } + + pub fn send_canvas_2d_msg(&self, msg: Canvas2dMsg) { + self.canvas_state.borrow().send_canvas_2d_msg(msg) + } + + pub fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg> { + self.canvas_state.borrow().ipc_renderer.clone() + } + + pub fn origin_is_clean(&self) -> bool { + self.canvas_state.borrow().origin_is_clean() + } + + pub fn get_rect(&self, rect: Rect<u32>) -> Vec<u8> { + self.canvas_state + .borrow() + .get_rect(self.canvas.as_ref().map(|c| &**c), rect) + } +} + +pub trait LayoutCanvasRenderingContext2DHelpers { + #[allow(unsafe_code)] + unsafe fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg>; + #[allow(unsafe_code)] + unsafe fn get_canvas_id(&self) -> CanvasId; +} + +impl LayoutCanvasRenderingContext2DHelpers for LayoutDom<CanvasRenderingContext2D> { + #[allow(unsafe_code)] + unsafe fn get_ipc_renderer(&self) -> IpcSender<CanvasMsg> { + (*self.unsafe_get()) + .canvas_state + .borrow_for_layout() + .ipc_renderer + .clone() + } + + #[allow(unsafe_code)] + unsafe fn get_canvas_id(&self) -> CanvasId { + (*self.unsafe_get()) + .canvas_state + .borrow_for_layout() + .get_canvas_id() + } +} + +// We add a guard to each of methods by the spec: +// http://www.w3.org/html/wg/drafts/2dcontext/html5_canvas_CR/ +// +// > Except where otherwise specified, for the 2D context interface, +// > any method call with a numeric argument whose value is infinite or a NaN value must be ignored. +// +// Restricted values are guarded in glue code. Therefore we need not add a guard. +// +// FIXME: this behavior should might be generated by some annotattions to idl. +impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { + // https://html.spec.whatwg.org/multipage/#dom-context-2d-canvas + fn Canvas(&self) -> DomRoot<HTMLCanvasElement> { + // This method is not called from a paint worklet rendering context, + // so it's OK to panic if self.canvas is None. + DomRoot::from_ref(self.canvas.as_ref().expect("No canvas.")) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-save + fn Save(&self) { + self.canvas_state.borrow().Save() + } + + #[allow(unrooted_must_root)] + // https://html.spec.whatwg.org/multipage/#dom-context-2d-restore + fn Restore(&self) { + self.canvas_state.borrow().Restore() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-scale + fn Scale(&self, x: f64, y: f64) { + self.canvas_state.borrow().Scale(x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-rotate + fn Rotate(&self, angle: f64) { + self.canvas_state.borrow().Rotate(angle) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-translate + fn Translate(&self, x: f64, y: f64) { + self.canvas_state.borrow().Translate(x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-transform + fn Transform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { + self.canvas_state.borrow().Transform(a, b, c, d, e, f) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-settransform + fn SetTransform(&self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) { + self.canvas_state.borrow().SetTransform(a, b, c, d, e, f) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-resettransform + fn ResetTransform(&self) { + self.canvas_state.borrow().ResetTransform() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha + fn GlobalAlpha(&self) -> f64 { + self.canvas_state.borrow().GlobalAlpha() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalalpha + fn SetGlobalAlpha(&self, alpha: f64) { + self.canvas_state.borrow().SetGlobalAlpha(alpha) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation + fn GlobalCompositeOperation(&self) -> DOMString { + self.canvas_state.borrow().GlobalCompositeOperation() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-globalcompositeoperation + fn SetGlobalCompositeOperation(&self, op_str: DOMString) { + self.canvas_state + .borrow() + .SetGlobalCompositeOperation(op_str) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-fillrect + fn FillRect(&self, x: f64, y: f64, width: f64, height: f64) { + self.canvas_state.borrow().FillRect(x, y, width, height); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-clearrect + fn ClearRect(&self, x: f64, y: f64, width: f64, height: f64) { + self.canvas_state.borrow().ClearRect(x, y, width, height); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokerect + fn StrokeRect(&self, x: f64, y: f64, width: f64, height: f64) { + self.canvas_state.borrow().StrokeRect(x, y, width, height); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-beginpath + fn BeginPath(&self) { + self.canvas_state.borrow().BeginPath() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-closepath + fn ClosePath(&self) { + self.canvas_state.borrow().ClosePath() + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-fill + fn Fill(&self, fill_rule: CanvasFillRule) { + self.canvas_state.borrow().Fill(fill_rule); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-stroke + fn Stroke(&self) { + self.canvas_state.borrow().Stroke(); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-clip + fn Clip(&self, fill_rule: CanvasFillRule) { + self.canvas_state.borrow().Clip(fill_rule) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-ispointinpath + fn IsPointInPath(&self, x: f64, y: f64, fill_rule: CanvasFillRule) -> bool { + self.canvas_state + .borrow() + .IsPointInPath(&self.global(), x, y, fill_rule) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-filltext + fn FillText(&self, text: DOMString, x: f64, y: f64, max_width: Option<f64>) { + self.canvas_state.borrow().FillText(text, x, y, max_width); + self.mark_as_dirty(); + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage + fn DrawImage(&self, image: CanvasImageSource, dx: f64, dy: f64) -> ErrorResult { + self.canvas_state + .borrow() + .DrawImage(self.canvas.as_ref().map(|c| &**c), image, dx, dy) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage + fn DrawImage_( + &self, + image: CanvasImageSource, + dx: f64, + dy: f64, + dw: f64, + dh: f64, + ) -> ErrorResult { + self.canvas_state.borrow().DrawImage_( + self.canvas.as_ref().map(|c| &**c), + image, + dx, + dy, + dw, + dh, + ) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-drawimage + fn DrawImage__( + &self, + image: CanvasImageSource, + sx: f64, + sy: f64, + sw: f64, + sh: f64, + dx: f64, + dy: f64, + dw: f64, + dh: f64, + ) -> ErrorResult { + self.canvas_state.borrow().DrawImage__( + self.canvas.as_ref().map(|c| &**c), + image, + sx, + sy, + sw, + sh, + dx, + dy, + dw, + dh, + ) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-moveto + fn MoveTo(&self, x: f64, y: f64) { + self.canvas_state.borrow().MoveTo(x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-lineto + fn LineTo(&self, x: f64, y: f64) { + self.canvas_state.borrow().LineTo(x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-rect + fn Rect(&self, x: f64, y: f64, width: f64, height: f64) { + self.canvas_state.borrow().Rect(x, y, width, height) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-quadraticcurveto + fn QuadraticCurveTo(&self, cpx: f64, cpy: f64, x: f64, y: f64) { + self.canvas_state.borrow().QuadraticCurveTo(cpx, cpy, x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-beziercurveto + fn BezierCurveTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, x: f64, y: f64) { + self.canvas_state + .borrow() + .BezierCurveTo(cp1x, cp1y, cp2x, cp2y, x, y) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-arc + fn Arc(&self, x: f64, y: f64, r: f64, start: f64, end: f64, ccw: bool) -> ErrorResult { + self.canvas_state.borrow().Arc(x, y, r, start, end, ccw) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-arcto + fn ArcTo(&self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, r: f64) -> ErrorResult { + self.canvas_state.borrow().ArcTo(cp1x, cp1y, cp2x, cp2y, r) + } + + // https://html.spec.whatwg.org/multipage/#dom-context-2d-ellipse + fn Ellipse( + &self, + x: f64, + y: f64, + rx: f64, + ry: f64, + rotation: f64, + start: f64, + end: f64, + ccw: bool, + ) -> ErrorResult { + self.canvas_state + .borrow() + .Ellipse(x, y, rx, ry, rotation, start, end, ccw) + } // https://html.spec.whatwg.org/multipage/#dom-context-2d-imagesmoothingenabled fn ImageSmoothingEnabled(&self) -> bool { - let state = self.state.borrow(); - state.image_smoothing_enabled + self.canvas_state.borrow().ImageSmoothingEnabled() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-imagesmoothingenabled fn SetImageSmoothingEnabled(&self, value: bool) { - self.state.borrow_mut().image_smoothing_enabled = value; + self.canvas_state.borrow().SetImageSmoothingEnabled(value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle fn StrokeStyle(&self) -> StringOrCanvasGradientOrCanvasPattern { - match self.state.borrow().stroke_style { - CanvasFillOrStrokeStyle::Color(ref rgba) => { - let mut result = String::new(); - serialize(rgba, &mut result).unwrap(); - StringOrCanvasGradientOrCanvasPattern::String(DOMString::from(result)) - }, - CanvasFillOrStrokeStyle::Gradient(ref gradient) => { - StringOrCanvasGradientOrCanvasPattern::CanvasGradient(DomRoot::from_ref(&*gradient)) - }, - CanvasFillOrStrokeStyle::Pattern(ref pattern) => { - StringOrCanvasGradientOrCanvasPattern::CanvasPattern(DomRoot::from_ref(&*pattern)) - }, - } + self.canvas_state.borrow().StrokeStyle() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle fn SetStrokeStyle(&self, value: StringOrCanvasGradientOrCanvasPattern) { - match value { - StringOrCanvasGradientOrCanvasPattern::String(string) => { - if let Ok(rgba) = self.parse_color(&string) { - self.state.borrow_mut().stroke_style = CanvasFillOrStrokeStyle::Color(rgba); - self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle(FillOrStrokeStyle::Color( - rgba, - ))); - } - }, - StringOrCanvasGradientOrCanvasPattern::CanvasGradient(gradient) => { - self.state.borrow_mut().stroke_style = - CanvasFillOrStrokeStyle::Gradient(Dom::from_ref(&*gradient)); - self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle( - gradient.to_fill_or_stroke_style(), - )); - }, - StringOrCanvasGradientOrCanvasPattern::CanvasPattern(pattern) => { - self.state.borrow_mut().stroke_style = - CanvasFillOrStrokeStyle::Pattern(Dom::from_ref(&*pattern)); - self.send_canvas_2d_msg(Canvas2dMsg::SetStrokeStyle( - pattern.to_fill_or_stroke_style(), - )); - if !pattern.origin_is_clean() { - self.set_origin_unclean(); - } - }, - } + self.canvas_state + .borrow() + .SetStrokeStyle(self.canvas.as_ref().map(|c| &**c), value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle fn FillStyle(&self) -> StringOrCanvasGradientOrCanvasPattern { - match self.state.borrow().fill_style { - CanvasFillOrStrokeStyle::Color(ref rgba) => { - let mut result = String::new(); - serialize(rgba, &mut result).unwrap(); - StringOrCanvasGradientOrCanvasPattern::String(DOMString::from(result)) - }, - CanvasFillOrStrokeStyle::Gradient(ref gradient) => { - StringOrCanvasGradientOrCanvasPattern::CanvasGradient(DomRoot::from_ref(&*gradient)) - }, - CanvasFillOrStrokeStyle::Pattern(ref pattern) => { - StringOrCanvasGradientOrCanvasPattern::CanvasPattern(DomRoot::from_ref(&*pattern)) - }, - } + self.canvas_state.borrow().FillStyle() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-strokestyle fn SetFillStyle(&self, value: StringOrCanvasGradientOrCanvasPattern) { - match value { - StringOrCanvasGradientOrCanvasPattern::String(string) => { - if let Ok(rgba) = self.parse_color(&string) { - self.state.borrow_mut().fill_style = CanvasFillOrStrokeStyle::Color(rgba); - self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle(FillOrStrokeStyle::Color( - rgba, - ))) - } - }, - StringOrCanvasGradientOrCanvasPattern::CanvasGradient(gradient) => { - self.state.borrow_mut().fill_style = - CanvasFillOrStrokeStyle::Gradient(Dom::from_ref(&*gradient)); - self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle( - gradient.to_fill_or_stroke_style(), - )); - }, - StringOrCanvasGradientOrCanvasPattern::CanvasPattern(pattern) => { - self.state.borrow_mut().fill_style = - CanvasFillOrStrokeStyle::Pattern(Dom::from_ref(&*pattern)); - self.send_canvas_2d_msg(Canvas2dMsg::SetFillStyle( - pattern.to_fill_or_stroke_style(), - )); - if !pattern.origin_is_clean() { - self.set_origin_unclean(); - } - }, - } + self.canvas_state + .borrow() + .SetFillStyle(self.canvas.as_ref().map(|c| &**c), value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-createimagedata fn CreateImageData(&self, sw: i32, sh: i32) -> Fallible<DomRoot<ImageData>> { - if sw == 0 || sh == 0 { - return Err(Error::IndexSize); - } - ImageData::new(&self.global(), sw.abs() as u32, sh.abs() as u32, None) + self.canvas_state + .borrow() + .CreateImageData(&self.global(), sw, sh) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-createimagedata fn CreateImageData_(&self, imagedata: &ImageData) -> Fallible<DomRoot<ImageData>> { - ImageData::new(&self.global(), imagedata.Width(), imagedata.Height(), None) + self.canvas_state + .borrow() + .CreateImageData_(&self.global(), imagedata) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-getimagedata fn GetImageData(&self, sx: i32, sy: i32, sw: i32, sh: i32) -> Fallible<DomRoot<ImageData>> { - // FIXME(nox): There are many arithmetic operations here that can - // overflow or underflow, this should probably be audited. - - if sw == 0 || sh == 0 { - return Err(Error::IndexSize); - } - - if !self.origin_is_clean() { - return Err(Error::Security); - } - - let (origin, size) = adjust_size_sign(Point2D::new(sx, sy), Size2D::new(sw, sh)); - // FIXME(nox): This is probably wrong when this is a context for an - // offscreen canvas. - let canvas_size = self - .canvas - .as_ref() - .map_or(Size2D::zero(), |c| c.get_size()); - let read_rect = match pixels::clip(origin, size, canvas_size) { - Some(rect) => rect, - None => { - // All the pixels are outside the canvas surface. - return ImageData::new(&self.global(), size.width, size.height, None); - }, - }; - - ImageData::new( + self.canvas_state.borrow().GetImageData( + self.canvas.as_ref().map(|c| &**c), &self.global(), - size.width, - size.height, - Some(self.get_rect(read_rect)), + sx, + sy, + sw, + sh, ) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-putimagedata fn PutImageData(&self, imagedata: &ImageData, dx: i32, dy: i32) { - self.PutImageData_( + self.canvas_state.borrow().PutImageData( + self.canvas.as_ref().map(|c| &**c), imagedata, dx, dy, - 0, - 0, - imagedata.Width() as i32, - imagedata.Height() as i32, ) } @@ -1239,52 +1857,16 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { dirty_width: i32, dirty_height: i32, ) { - // FIXME(nox): There are many arithmetic operations here that can - // overflow or underflow, this should probably be audited. - - let imagedata_size = Size2D::new(imagedata.Width(), imagedata.Height()); - if imagedata_size.area() == 0 { - return; - } - - // Step 1. - // Done later. - - // Step 2. - // TODO: throw InvalidState if buffer is detached. - - // FIXME(nox): This is probably wrong when this is a context for an - // offscreen canvas. - let canvas_size = self - .canvas - .as_ref() - .map_or(Size2D::zero(), |c| c.get_size()); - - // Steps 3-6. - let (src_origin, src_size) = adjust_size_sign( - Point2D::new(dirty_x, dirty_y), - Size2D::new(dirty_width, dirty_height), - ); - let src_rect = match pixels::clip(src_origin, src_size, imagedata_size) { - Some(rect) => rect, - None => return, - }; - let (dst_origin, _) = adjust_size_sign( - Point2D::new(dirty_x.saturating_add(dx), dirty_y.saturating_add(dy)), - Size2D::new(dirty_width, dirty_height), + self.canvas_state.borrow().PutImageData_( + self.canvas.as_ref().map(|c| &**c), + imagedata, + dx, + dy, + dirty_x, + dirty_y, + dirty_width, + dirty_height, ); - // By clipping to the canvas surface, we avoid sending any pixel - // that would fall outside it. - let dst_rect = match pixels::clip(dst_origin, src_rect.size, canvas_size) { - Some(rect) => rect, - None => return, - }; - - // Step 7. - let (sender, receiver) = ipc::bytes_channel().unwrap(); - let pixels = unsafe { &imagedata.get_rect(Rect::new(src_rect.origin, dst_rect.size)) }; - self.send_canvas_2d_msg(Canvas2dMsg::PutImageData(dst_rect, receiver)); - sender.send(pixels).unwrap(); self.mark_as_dirty(); } @@ -1296,10 +1878,9 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { x1: Finite<f64>, y1: Finite<f64>, ) -> DomRoot<CanvasGradient> { - CanvasGradient::new( - &self.global(), - CanvasGradientStyle::Linear(LinearGradientStyle::new(*x0, *y0, *x1, *y1, Vec::new())), - ) + self.canvas_state + .borrow() + .CreateLinearGradient(&self.global(), x0, y0, x1, y1) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-createradialgradient @@ -1312,197 +1893,100 @@ impl CanvasRenderingContext2DMethods for CanvasRenderingContext2D { y1: Finite<f64>, r1: Finite<f64>, ) -> Fallible<DomRoot<CanvasGradient>> { - if *r0 < 0. || *r1 < 0. { - return Err(Error::IndexSize); - } - - Ok(CanvasGradient::new( - &self.global(), - CanvasGradientStyle::Radial(RadialGradientStyle::new( - *x0, - *y0, - *r0, - *x1, - *y1, - *r1, - Vec::new(), - )), - )) + self.canvas_state + .borrow() + .CreateRadialGradient(&self.global(), x0, y0, r0, x1, y1, r1) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-createpattern fn CreatePattern( &self, image: CanvasImageSource, - mut repetition: DOMString, + repetition: DOMString, ) -> Fallible<DomRoot<CanvasPattern>> { - let (image_data, image_size) = match image { - CanvasImageSource::HTMLImageElement(ref image) => { - // https://html.spec.whatwg.org/multipage/#img-error - // If the image argument is an HTMLImageElement object that is in the broken state, - // then throw an InvalidStateError exception - image - .get_url() - .and_then(|url| self.fetch_image_data(url)) - .ok_or(Error::InvalidState)? - }, - CanvasImageSource::HTMLCanvasElement(ref canvas) => { - let (data, size) = canvas.fetch_all_data().ok_or(Error::InvalidState)?; - let data = data - .map(|data| data.to_vec()) - .unwrap_or_else(|| vec![0; size.area() as usize * 4]); - (data, size) - }, - CanvasImageSource::CSSStyleValue(ref value) => value - .get_url(self.base_url.clone()) - .and_then(|url| self.fetch_image_data(url)) - .ok_or(Error::InvalidState)?, - }; - - if repetition.is_empty() { - repetition.push_str("repeat"); - } - - if let Ok(rep) = RepetitionStyle::from_str(&repetition) { - Ok(CanvasPattern::new( - &self.global(), - image_data, - image_size, - rep, - self.is_origin_clean(image), - )) - } else { - Err(Error::Syntax) - } + self.canvas_state + .borrow() + .CreatePattern(&self.global(), image, repetition) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linewidth fn LineWidth(&self) -> f64 { - let state = self.state.borrow(); - state.line_width + self.canvas_state.borrow().LineWidth() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linewidth fn SetLineWidth(&self, width: f64) { - if !width.is_finite() || width <= 0.0 { - return; - } - - self.state.borrow_mut().line_width = width; - self.send_canvas_2d_msg(Canvas2dMsg::SetLineWidth(width as f32)) + self.canvas_state.borrow().SetLineWidth(width) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linecap fn LineCap(&self) -> CanvasLineCap { - match self.state.borrow().line_cap { - LineCapStyle::Butt => CanvasLineCap::Butt, - LineCapStyle::Round => CanvasLineCap::Round, - LineCapStyle::Square => CanvasLineCap::Square, - } + self.canvas_state.borrow().LineCap() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linecap fn SetLineCap(&self, cap: CanvasLineCap) { - let line_cap = match cap { - CanvasLineCap::Butt => LineCapStyle::Butt, - CanvasLineCap::Round => LineCapStyle::Round, - CanvasLineCap::Square => LineCapStyle::Square, - }; - self.state.borrow_mut().line_cap = line_cap; - self.send_canvas_2d_msg(Canvas2dMsg::SetLineCap(line_cap)); + self.canvas_state.borrow().SetLineCap(cap) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linejoin fn LineJoin(&self) -> CanvasLineJoin { - match self.state.borrow().line_join { - LineJoinStyle::Round => CanvasLineJoin::Round, - LineJoinStyle::Bevel => CanvasLineJoin::Bevel, - LineJoinStyle::Miter => CanvasLineJoin::Miter, - } + self.canvas_state.borrow().LineJoin() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-linejoin fn SetLineJoin(&self, join: CanvasLineJoin) { - let line_join = match join { - CanvasLineJoin::Round => LineJoinStyle::Round, - CanvasLineJoin::Bevel => LineJoinStyle::Bevel, - CanvasLineJoin::Miter => LineJoinStyle::Miter, - }; - self.state.borrow_mut().line_join = line_join; - self.send_canvas_2d_msg(Canvas2dMsg::SetLineJoin(line_join)); + self.canvas_state.borrow().SetLineJoin(join) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-miterlimit fn MiterLimit(&self) -> f64 { - let state = self.state.borrow(); - state.miter_limit + self.canvas_state.borrow().MiterLimit() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-miterlimit fn SetMiterLimit(&self, limit: f64) { - if !limit.is_finite() || limit <= 0.0 { - return; - } - - self.state.borrow_mut().miter_limit = limit; - self.send_canvas_2d_msg(Canvas2dMsg::SetMiterLimit(limit as f32)) + self.canvas_state.borrow().SetMiterLimit(limit) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsetx fn ShadowOffsetX(&self) -> f64 { - self.state.borrow().shadow_offset_x + self.canvas_state.borrow().ShadowOffsetX() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsetx fn SetShadowOffsetX(&self, value: f64) { - if !value.is_finite() || value == self.state.borrow().shadow_offset_x { - return; - } - self.state.borrow_mut().shadow_offset_x = value; - self.send_canvas_2d_msg(Canvas2dMsg::SetShadowOffsetX(value)) + self.canvas_state.borrow().SetShadowOffsetX(value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsety fn ShadowOffsetY(&self) -> f64 { - self.state.borrow().shadow_offset_y + self.canvas_state.borrow().ShadowOffsetY() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowoffsety fn SetShadowOffsetY(&self, value: f64) { - if !value.is_finite() || value == self.state.borrow().shadow_offset_y { - return; - } - self.state.borrow_mut().shadow_offset_y = value; - self.send_canvas_2d_msg(Canvas2dMsg::SetShadowOffsetY(value)) + self.canvas_state.borrow().SetShadowOffsetY(value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowblur fn ShadowBlur(&self) -> f64 { - self.state.borrow().shadow_blur + self.canvas_state.borrow().ShadowBlur() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowblur fn SetShadowBlur(&self, value: f64) { - if !value.is_finite() || value < 0f64 || value == self.state.borrow().shadow_blur { - return; - } - self.state.borrow_mut().shadow_blur = value; - self.send_canvas_2d_msg(Canvas2dMsg::SetShadowBlur(value)) + self.canvas_state.borrow().SetShadowBlur(value) } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowcolor fn ShadowColor(&self) -> DOMString { - let mut result = String::new(); - serialize(&self.state.borrow().shadow_color, &mut result).unwrap(); - DOMString::from(result) + self.canvas_state.borrow().ShadowColor() } // https://html.spec.whatwg.org/multipage/#dom-context-2d-shadowcolor fn SetShadowColor(&self, value: DOMString) { - if let Ok(color) = parse_color(&value) { - self.state.borrow_mut().shadow_color = color; - self.send_canvas_2d_msg(Canvas2dMsg::SetShadowColor(color)) - } + self.canvas_state.borrow().SetShadowColor(value) } } |