mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
311 lines
13 KiB
Rust
311 lines
13 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 dom::bindings::codegen::Bindings::CanvasRenderingContext2DBinding;
|
|
use dom::bindings::codegen::Bindings::CanvasRenderingContext2DBinding::CanvasRenderingContext2DMethods;
|
|
use dom::bindings::codegen::Bindings::CanvasRenderingContext2DBinding::CanvasWindingRule;
|
|
use dom::bindings::codegen::Bindings::ImageDataBinding::ImageDataMethods;
|
|
use dom::bindings::codegen::UnionTypes::StringOrCanvasGradientOrCanvasPattern;
|
|
use dom::bindings::error::Error::{IndexSize, TypeError};
|
|
use dom::bindings::error::Fallible;
|
|
use dom::bindings::global::{GlobalRef, GlobalField};
|
|
use dom::bindings::js::{JS, JSRef, LayoutJS, Temporary};
|
|
use dom::bindings::utils::{Reflector, reflect_dom_object};
|
|
use dom::canvasgradient::{CanvasGradient, CanvasGradientStyle, ToFillOrStrokeStyle};
|
|
use dom::htmlcanvaselement::{HTMLCanvasElement, HTMLCanvasElementHelpers};
|
|
use dom::imagedata::{ImageData, ImageDataHelpers};
|
|
|
|
use cssparser::Color as CSSColor;
|
|
use cssparser::{Parser, RGBA, ToCss};
|
|
use geom::matrix2d::Matrix2D;
|
|
use geom::point::Point2D;
|
|
use geom::rect::Rect;
|
|
use geom::size::Size2D;
|
|
|
|
use canvas::canvas_paint_task::{CanvasMsg, CanvasPaintTask, FillOrStrokeStyle};
|
|
use canvas::canvas_paint_task::{LinearGradientStyle, RadialGradientStyle};
|
|
|
|
use std::borrow::ToOwned;
|
|
use std::cell::Cell;
|
|
use std::num::{Float, ToPrimitive};
|
|
use std::sync::mpsc::{channel, Sender};
|
|
|
|
#[dom_struct]
|
|
pub struct CanvasRenderingContext2D {
|
|
reflector_: Reflector,
|
|
global: GlobalField,
|
|
renderer: Sender<CanvasMsg>,
|
|
canvas: JS<HTMLCanvasElement>,
|
|
stroke_color: Cell<RGBA>,
|
|
fill_color: Cell<RGBA>,
|
|
transform: Cell<Matrix2D<f32>>,
|
|
}
|
|
|
|
impl CanvasRenderingContext2D {
|
|
fn new_inherited(global: GlobalRef, canvas: JSRef<HTMLCanvasElement>, size: Size2D<i32>)
|
|
-> CanvasRenderingContext2D {
|
|
let black = RGBA {
|
|
red: 0.0,
|
|
green: 0.0,
|
|
blue: 0.0,
|
|
alpha: 1.0,
|
|
};
|
|
CanvasRenderingContext2D {
|
|
reflector_: Reflector::new(),
|
|
global: GlobalField::from_rooted(&global),
|
|
renderer: CanvasPaintTask::start(size),
|
|
canvas: JS::from_rooted(canvas),
|
|
stroke_color: Cell::new(black),
|
|
fill_color: Cell::new(black),
|
|
transform: Cell::new(Matrix2D::identity()),
|
|
}
|
|
}
|
|
|
|
pub fn new(global: GlobalRef, canvas: JSRef<HTMLCanvasElement>, size: Size2D<i32>)
|
|
-> Temporary<CanvasRenderingContext2D> {
|
|
reflect_dom_object(box CanvasRenderingContext2D::new_inherited(global, canvas, size),
|
|
global, CanvasRenderingContext2DBinding::Wrap)
|
|
}
|
|
|
|
pub fn recreate(&self, size: Size2D<i32>) {
|
|
self.renderer.send(CanvasMsg::Recreate(size)).unwrap();
|
|
}
|
|
|
|
fn update_transform(&self) {
|
|
self.renderer.send(CanvasMsg::SetTransform(self.transform.get())).unwrap()
|
|
}
|
|
}
|
|
|
|
pub trait LayoutCanvasRenderingContext2DHelpers {
|
|
#[allow(unsafe_code)]
|
|
unsafe fn get_renderer(&self) -> Sender<CanvasMsg>;
|
|
}
|
|
|
|
impl LayoutCanvasRenderingContext2DHelpers for LayoutJS<CanvasRenderingContext2D> {
|
|
#[allow(unsafe_code)]
|
|
unsafe fn get_renderer(&self) -> Sender<CanvasMsg> {
|
|
(*self.unsafe_get()).renderer.clone()
|
|
}
|
|
}
|
|
|
|
impl<'a> CanvasRenderingContext2DMethods for JSRef<'a, CanvasRenderingContext2D> {
|
|
fn Canvas(self) -> Temporary<HTMLCanvasElement> {
|
|
Temporary::new(self.canvas)
|
|
}
|
|
|
|
fn Scale(self, x: f64, y: f64) {
|
|
self.transform.set(self.transform.get().scale(x as f32, y as f32));
|
|
self.update_transform()
|
|
}
|
|
|
|
fn Translate(self, x: f64, y: f64) {
|
|
self.transform.set(self.transform.get().translate(x as f32, y as f32));
|
|
self.update_transform()
|
|
}
|
|
|
|
fn Transform(self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) {
|
|
self.transform.set(self.transform.get().mul(&Matrix2D::new(a as f32,
|
|
b as f32,
|
|
c as f32,
|
|
d as f32,
|
|
e as f32,
|
|
f as f32)));
|
|
self.update_transform()
|
|
}
|
|
|
|
fn SetTransform(self, a: f64, b: f64, c: f64, d: f64, e: f64, f: f64) {
|
|
self.transform.set(Matrix2D::new(a as f32,
|
|
b as f32,
|
|
c as f32,
|
|
d as f32,
|
|
e as f32,
|
|
f as f32));
|
|
self.update_transform()
|
|
}
|
|
|
|
fn FillRect(self, x: f64, y: f64, width: f64, height: f64) {
|
|
let rect = Rect(Point2D(x as f32, y as f32), Size2D(width as f32, height as f32));
|
|
self.renderer.send(CanvasMsg::FillRect(rect)).unwrap();
|
|
}
|
|
|
|
fn ClearRect(self, x: f64, y: f64, width: f64, height: f64) {
|
|
let rect = Rect(Point2D(x as f32, y as f32), Size2D(width as f32, height as f32));
|
|
self.renderer.send(CanvasMsg::ClearRect(rect)).unwrap();
|
|
}
|
|
|
|
fn StrokeRect(self, x: f64, y: f64, width: f64, height: f64) {
|
|
let rect = Rect(Point2D(x as f32, y as f32), Size2D(width as f32, height as f32));
|
|
self.renderer.send(CanvasMsg::StrokeRect(rect)).unwrap();
|
|
}
|
|
|
|
fn BeginPath(self) {
|
|
self.renderer.send(CanvasMsg::BeginPath).unwrap();
|
|
}
|
|
|
|
fn ClosePath(self) {
|
|
self.renderer.send(CanvasMsg::ClosePath).unwrap();
|
|
}
|
|
|
|
fn Fill(self, _: CanvasWindingRule) {
|
|
self.renderer.send(CanvasMsg::Fill).unwrap();
|
|
}
|
|
|
|
fn MoveTo(self, x: f64, y: f64) {
|
|
self.renderer.send(CanvasMsg::MoveTo(Point2D(x as f32, y as f32))).unwrap();
|
|
}
|
|
|
|
fn LineTo(self, x: f64, y: f64) {
|
|
self.renderer.send(CanvasMsg::LineTo(Point2D(x as f32, y as f32))).unwrap();
|
|
}
|
|
|
|
fn QuadraticCurveTo(self, cpx: f64, cpy: f64, x: f64, y: f64) {
|
|
self.renderer.send(CanvasMsg::QuadraticCurveTo(Point2D(cpx as f32, cpy as f32),
|
|
Point2D(x as f32, y as f32))).unwrap();
|
|
}
|
|
|
|
fn BezierCurveTo(self, cp1x: f64, cp1y: f64, cp2x: f64, cp2y: f64, x: f64, y: f64) {
|
|
self.renderer.send(CanvasMsg::BezierCurveTo(Point2D(cp1x as f32, cp1y as f32),
|
|
Point2D(cp2x as f32, cp2y as f32),
|
|
Point2D(x as f32, y as f32))).unwrap();
|
|
}
|
|
|
|
fn Arc(self, x: f64, y: f64, r: f64, start: f64, end: f64, ccw: bool) {
|
|
self.renderer.send(CanvasMsg::Arc(Point2D(x as f32, y as f32), r as f32,
|
|
start as f32, end as f32, ccw)).unwrap();
|
|
}
|
|
|
|
fn StrokeStyle(self) -> StringOrCanvasGradientOrCanvasPattern {
|
|
// FIXME(pcwalton, #4761): This is not spec-compliant. See:
|
|
//
|
|
// https://html.spec.whatwg.org/multipage/scripting.html#serialisation-of-a-colour
|
|
let mut result = String::new();
|
|
self.stroke_color.get().to_css(&mut result).unwrap();
|
|
StringOrCanvasGradientOrCanvasPattern::eString(result)
|
|
}
|
|
|
|
fn SetStrokeStyle(self, value: StringOrCanvasGradientOrCanvasPattern) {
|
|
match value {
|
|
StringOrCanvasGradientOrCanvasPattern::eString(string) => {
|
|
match parse_color(string.as_slice()) {
|
|
Ok(rgba) => {
|
|
self.stroke_color.set(rgba);
|
|
self.renderer
|
|
.send(CanvasMsg::SetStrokeStyle(FillOrStrokeStyle::Color(rgba)))
|
|
.unwrap();
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
_ => {
|
|
// TODO(pcwalton)
|
|
}
|
|
}
|
|
}
|
|
|
|
fn FillStyle(self) -> StringOrCanvasGradientOrCanvasPattern {
|
|
// FIXME(pcwalton, #4761): This is not spec-compliant. See:
|
|
//
|
|
// https://html.spec.whatwg.org/multipage/scripting.html#serialisation-of-a-colour
|
|
let mut result = String::new();
|
|
self.stroke_color.get().to_css(&mut result).unwrap();
|
|
StringOrCanvasGradientOrCanvasPattern::eString(result)
|
|
}
|
|
|
|
fn SetFillStyle(self, value: StringOrCanvasGradientOrCanvasPattern) {
|
|
match value {
|
|
StringOrCanvasGradientOrCanvasPattern::eString(string) => {
|
|
match parse_color(string.as_slice()) {
|
|
Ok(rgba) => {
|
|
self.fill_color.set(rgba);
|
|
self.renderer
|
|
.send(CanvasMsg::SetFillStyle(FillOrStrokeStyle::Color(rgba)))
|
|
.unwrap()
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
StringOrCanvasGradientOrCanvasPattern::eCanvasGradient(gradient) => {
|
|
self.renderer.send(CanvasMsg::SetFillStyle(gradient.root().r().to_fill_or_stroke_style())).unwrap();
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
fn CreateImageData(self, sw: f64, sh: f64) -> Fallible<Temporary<ImageData>> {
|
|
if sw == 0.0 || sh == 0.0 {
|
|
return Err(IndexSize)
|
|
}
|
|
|
|
Ok(ImageData::new(self.global.root().r(), sw.abs().to_u32().unwrap(), sh.abs().to_u32().unwrap(), None))
|
|
}
|
|
|
|
fn CreateImageData_(self, imagedata: JSRef<ImageData>) -> Fallible<Temporary<ImageData>> {
|
|
Ok(ImageData::new(self.global.root().r(), imagedata.Width(), imagedata.Height(), None))
|
|
}
|
|
|
|
fn GetImageData(self, sx: f64, sy: f64, sw: f64, sh: f64) -> Fallible<Temporary<ImageData>> {
|
|
if sw == 0.0 || sh == 0.0 {
|
|
return Err(IndexSize)
|
|
}
|
|
|
|
let (sender, receiver) = channel::<Vec<u8>>();
|
|
let dest_rect = Rect(Point2D(sx.to_i32().unwrap(), sy.to_i32().unwrap()), Size2D(sw.to_i32().unwrap(), sh.to_i32().unwrap()));
|
|
let canvas_size = self.canvas.root().r().get_size();
|
|
self.renderer.send(CanvasMsg::GetImageData(dest_rect, canvas_size, sender)).unwrap();
|
|
let data = receiver.recv().unwrap();
|
|
Ok(ImageData::new(self.global.root().r(), sw.abs().to_u32().unwrap(), sh.abs().to_u32().unwrap(), Some(data)))
|
|
}
|
|
|
|
fn PutImageData(self, imagedata: JSRef<ImageData>, dx: f64, dy: f64) {
|
|
let data = imagedata.get_data_array(&self.global.root().r());
|
|
let image_data_rect = Rect(Point2D(dx.to_i32().unwrap(), dy.to_i32().unwrap()), imagedata.get_size());
|
|
let dirty_rect = None;
|
|
let canvas_size = self.canvas.root().r().get_size();
|
|
self.renderer.send(CanvasMsg::PutImageData(data, image_data_rect, dirty_rect, canvas_size)).unwrap()
|
|
}
|
|
|
|
fn PutImageData_(self, imagedata: JSRef<ImageData>, dx: f64, dy: f64,
|
|
dirtyX: f64, dirtyY: f64, dirtyWidth: f64, dirtyHeight: f64) {
|
|
let data = imagedata.get_data_array(&self.global.root().r());
|
|
let image_data_rect = Rect(Point2D(dx.to_i32().unwrap(), dy.to_i32().unwrap()),
|
|
Size2D(imagedata.Width().to_i32().unwrap(),
|
|
imagedata.Height().to_i32().unwrap()));
|
|
let dirty_rect = Some(Rect(Point2D(dirtyX.to_i32().unwrap(), dirtyY.to_i32().unwrap()),
|
|
Size2D(dirtyWidth.to_i32().unwrap(),
|
|
dirtyHeight.to_i32().unwrap())));
|
|
let canvas_size = self.canvas.root().r().get_size();
|
|
self.renderer.send(CanvasMsg::PutImageData(data, image_data_rect, dirty_rect, canvas_size)).unwrap()
|
|
}
|
|
|
|
fn CreateLinearGradient(self, x0: f64, y0: f64, x1: f64, y1: f64) -> Fallible<Temporary<CanvasGradient>> {
|
|
if [x0, y0, x1, y1].iter().any(|x| x.is_nan() || x.is_infinite()) {
|
|
return Err(TypeError("One of the arguments of createLinearGradient() is not a finite floating-point value.".to_owned()));
|
|
}
|
|
Ok(CanvasGradient::new(self.global.root().r(),
|
|
CanvasGradientStyle::Linear(LinearGradientStyle::new(x0, y0, x1, y1, Vec::new()))))
|
|
}
|
|
|
|
fn CreateRadialGradient(self, x0: f64, y0: f64, r0: f64, x1: f64, y1: f64, r1: f64) -> Fallible<Temporary<CanvasGradient>> {
|
|
if [x0, y0, r0, x1, y1, r1].iter().any(|x| x.is_nan() || x.is_infinite()) {
|
|
return Err(TypeError("One of the arguments of createRadialGradient() is not a finite floating-point value.".to_owned()));
|
|
}
|
|
Ok(CanvasGradient::new(self.global.root().r(),
|
|
CanvasGradientStyle::Radial(RadialGradientStyle::new(x0, y0, r0, x1, y1, r1, Vec::new()))))
|
|
}
|
|
}
|
|
|
|
#[unsafe_destructor]
|
|
impl Drop for CanvasRenderingContext2D {
|
|
fn drop(&mut self) {
|
|
self.renderer.send(CanvasMsg::Close).unwrap();
|
|
}
|
|
}
|
|
|
|
pub fn parse_color(string: &str) -> Result<RGBA,()> {
|
|
match CSSColor::parse(&mut Parser::new(string.as_slice())) {
|
|
Ok(CSSColor::RGBA(rgba)) => Ok(rgba),
|
|
_ => Err(()),
|
|
}
|
|
}
|
|
|