/* 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/. */ #![crate_name = "canvas_traits"] #![crate_type = "rlib"] #![feature(core)] extern crate core; extern crate azure; extern crate geom; extern crate cssparser; extern crate gfx_traits; extern crate layers; extern crate offscreen_gl_context; use azure::azure::AzFloat; use azure::azure_hl::{DrawTarget, Pattern, ColorPattern}; use azure::azure_hl::{GradientStop, LinearGradientPattern, RadialGradientPattern, ExtendMode}; use azure::azure_hl::{JoinStyle, CapStyle, CompositionOp}; use cssparser::RGBA; use geom::matrix2d::Matrix2D; use geom::point::Point2D; use geom::rect::Rect; use geom::size::Size2D; use gfx_traits::color; use std::sync::mpsc::{Sender}; use layers::platform::surface::NativeSurface; use offscreen_gl_context::GLContextAttributes; use core::nonzero::NonZero; #[derive(Clone)] pub enum CanvasMsg { Canvas2d(Canvas2dMsg), Common(CanvasCommonMsg), WebGL(CanvasWebGLMsg), } #[derive(Clone)] pub enum Canvas2dMsg { Arc(Point2D, f32, f32, f32, bool), ArcTo(Point2D, Point2D, f32), DrawImage(Vec, Size2D, Rect, Rect, bool), DrawImageSelf(Size2D, Rect, Rect, bool), BeginPath, BezierCurveTo(Point2D, Point2D, Point2D), ClearRect(Rect), Clip, ClosePath, Fill, FillRect(Rect), GetImageData(Rect, Size2D, Sender>), LineTo(Point2D), MoveTo(Point2D), PutImageData(Vec, Rect, Option>), QuadraticCurveTo(Point2D, Point2D), Rect(Rect), RestoreContext, SaveContext, StrokeRect(Rect), Stroke, SetFillStyle(FillOrStrokeStyle), SetStrokeStyle(FillOrStrokeStyle), SetLineWidth(f32), SetLineCap(LineCapStyle), SetLineJoin(LineJoinStyle), SetMiterLimit(f32), SetGlobalAlpha(f32), SetGlobalComposition(CompositionOrBlending), SetTransform(Matrix2D), } #[derive(Clone)] pub enum CanvasWebGLMsg { GetContextAttributes(Sender), AttachShader(u32, u32), BufferData(u32, Vec, u32), Clear(u32), ClearColor(f32, f32, f32, f32), CompileShader(u32), CreateBuffer(Sender>>), CreateFramebuffer(Sender>>), CreateRenderbuffer(Sender>>), CreateTexture(Sender>>), CreateProgram(Sender>>), CreateShader(u32, Sender>>), DeleteBuffer(u32), DeleteFramebuffer(u32), DeleteRenderbuffer(u32), DeleteTexture(u32), DeleteProgram(u32), DeleteShader(u32), BindBuffer(u32, u32), BindFramebuffer(u32, u32), BindRenderbuffer(u32, u32), BindTexture(u32, u32), DrawArrays(u32, i32, i32), EnableVertexAttribArray(u32), GetAttribLocation(u32, String, Sender), GetShaderInfoLog(u32, Sender), GetShaderParameter(u32, u32, Sender), GetUniformLocation(u32, String, Sender>), LinkProgram(u32), ShaderSource(u32, String), Uniform4fv(i32, Vec), UseProgram(u32), VertexAttribPointer2f(u32, i32, bool, i32, i64), Viewport(i32, i32, i32, i32), DrawingBufferWidth(Sender), DrawingBufferHeight(Sender), } #[derive(Clone)] pub enum CanvasCommonMsg { Close, Recreate(Size2D), SendPixelContents(Sender>), SendNativeSurface(Sender), } #[derive(Clone)] pub struct CanvasGradientStop { pub offset: f64, pub color: RGBA, } #[derive(Clone)] pub struct LinearGradientStyle { pub x0: f64, pub y0: f64, pub x1: f64, pub y1: f64, pub stops: Vec } impl LinearGradientStyle { pub fn new(x0: f64, y0: f64, x1: f64, y1: f64, stops: Vec) -> LinearGradientStyle { LinearGradientStyle { x0: x0, y0: y0, x1: x1, y1: y1, stops: stops, } } } #[derive(Clone)] pub struct RadialGradientStyle { pub x0: f64, pub y0: f64, pub r0: f64, pub x1: f64, pub y1: f64, pub r1: f64, pub stops: Vec } impl RadialGradientStyle { pub fn new(x0: f64, y0: f64, r0: f64, x1: f64, y1: f64, r1: f64, stops: Vec) -> RadialGradientStyle { RadialGradientStyle { x0: x0, y0: y0, r0: r0, x1: x1, y1: y1, r1: r1, stops: stops, } } } #[derive(Clone)] pub enum FillOrStrokeStyle { Color(RGBA), LinearGradient(LinearGradientStyle), RadialGradient(RadialGradientStyle), } impl FillOrStrokeStyle { pub fn to_azure_pattern(&self, drawtarget: &DrawTarget) -> Pattern { match *self { FillOrStrokeStyle::Color(ref color) => { Pattern::Color(ColorPattern::new(color::new(color.red, color.green, color.blue, color.alpha))) }, FillOrStrokeStyle::LinearGradient(ref linear_gradient_style) => { let gradient_stops: Vec = linear_gradient_style.stops.iter().map(|s| { GradientStop { offset: s.offset as AzFloat, color: color::new(s.color.red, s.color.green, s.color.blue, s.color.alpha) } }).collect(); Pattern::LinearGradient(LinearGradientPattern::new( &Point2D(linear_gradient_style.x0 as AzFloat, linear_gradient_style.y0 as AzFloat), &Point2D(linear_gradient_style.x1 as AzFloat, linear_gradient_style.y1 as AzFloat), drawtarget.create_gradient_stops(&gradient_stops, ExtendMode::Clamp), &Matrix2D::identity())) }, FillOrStrokeStyle::RadialGradient(ref radial_gradient_style) => { let gradient_stops: Vec = radial_gradient_style.stops.iter().map(|s| { GradientStop { offset: s.offset as AzFloat, color: color::new(s.color.red, s.color.green, s.color.blue, s.color.alpha) } }).collect(); Pattern::RadialGradient(RadialGradientPattern::new( &Point2D(radial_gradient_style.x0 as AzFloat, radial_gradient_style.y0 as AzFloat), &Point2D(radial_gradient_style.x1 as AzFloat, radial_gradient_style.y1 as AzFloat), radial_gradient_style.r0 as AzFloat, radial_gradient_style.r1 as AzFloat, drawtarget.create_gradient_stops(&gradient_stops, ExtendMode::Clamp), &Matrix2D::identity())) } } } } #[derive(Copy, Clone, PartialEq)] pub enum LineCapStyle { Butt = 0, Round = 1, Square = 2, } impl LineCapStyle { pub fn to_azure_style(&self) -> CapStyle { match *self { LineCapStyle::Butt => CapStyle::Butt, LineCapStyle::Round => CapStyle::Round, LineCapStyle::Square => CapStyle::Square, } } pub fn from_str(string: &str) -> Option { match string { "butt" => Some(LineCapStyle::Butt), "round" => Some(LineCapStyle::Round), "square" => Some(LineCapStyle::Square), _ => None } } } #[derive(Copy, Clone, PartialEq)] pub enum LineJoinStyle { Round = 0, Bevel = 1, Miter = 2, } impl LineJoinStyle { pub fn to_azure_style(&self) -> JoinStyle { match *self { LineJoinStyle::Round => JoinStyle::Round, LineJoinStyle::Bevel => JoinStyle::Bevel, LineJoinStyle::Miter => JoinStyle::Miter, } } pub fn from_str(string: &str) -> Option { match string { "round" => Some(LineJoinStyle::Round), "bevel" => Some(LineJoinStyle::Bevel), "miter" => Some(LineJoinStyle::Miter), _ => None } } } #[derive(Copy, Clone, PartialEq)] pub enum CompositionStyle { SrcIn, SrcOut, SrcOver, SrcAtop, DestIn, DestOut, DestOver, DestAtop, Copy, Lighter, Xor, } impl CompositionStyle { pub fn to_azure_style(&self) -> CompositionOp { match *self { CompositionStyle::SrcIn => CompositionOp::In, CompositionStyle::SrcOut => CompositionOp::Out, CompositionStyle::SrcOver => CompositionOp::Over, CompositionStyle::SrcAtop => CompositionOp::Atop, CompositionStyle::DestIn => CompositionOp::DestIn, CompositionStyle::DestOut => CompositionOp::DestOut, CompositionStyle::DestOver => CompositionOp::DestOver, CompositionStyle::DestAtop => CompositionOp::DestAtop, CompositionStyle::Copy => CompositionOp::Source, CompositionStyle::Lighter => CompositionOp::Add, CompositionStyle::Xor => CompositionOp::Xor, } } pub fn from_str(string: &str) -> Option { match string { "source-in" => Some(CompositionStyle::SrcIn), "source-out" => Some(CompositionStyle::SrcOut), "source-over" => Some(CompositionStyle::SrcOver), "source-atop" => Some(CompositionStyle::SrcAtop), "destination-in" => Some(CompositionStyle::DestIn), "destination-out" => Some(CompositionStyle::DestOut), "destination-over" => Some(CompositionStyle::DestOver), "destination-atop" => Some(CompositionStyle::DestAtop), "copy" => Some(CompositionStyle::Copy), "lighter" => Some(CompositionStyle::Lighter), "xor" => Some(CompositionStyle::Xor), _ => None } } pub fn to_str(&self) -> &str { match *self { CompositionStyle::SrcIn => "source-in", CompositionStyle::SrcOut => "source-out", CompositionStyle::SrcOver => "source-over", CompositionStyle::SrcAtop => "source-atop", CompositionStyle::DestIn => "destination-in", CompositionStyle::DestOut => "destination-out", CompositionStyle::DestOver => "destination-over", CompositionStyle::DestAtop => "destination-atop", CompositionStyle::Copy => "copy", CompositionStyle::Lighter => "lighter", CompositionStyle::Xor => "xor", } } } #[derive(Copy, Clone, PartialEq)] pub enum BlendingStyle { Multiply, Screen, Overlay, Darken, Lighten, ColorDodge, ColorBurn, HardLight, SoftLight, Difference, Exclusion, Hue, Saturation, Color, Luminosity, } impl BlendingStyle { pub fn to_azure_style(&self) -> CompositionOp { match *self { BlendingStyle::Multiply => CompositionOp::Multiply, BlendingStyle::Screen => CompositionOp::Screen, BlendingStyle::Overlay => CompositionOp::Overlay, BlendingStyle::Darken => CompositionOp::Darken, BlendingStyle::Lighten => CompositionOp::Lighten, BlendingStyle::ColorDodge => CompositionOp::ColorDodge, BlendingStyle::ColorBurn => CompositionOp::ColorBurn, BlendingStyle::HardLight => CompositionOp::HardLight, BlendingStyle::SoftLight => CompositionOp::SoftLight, BlendingStyle::Difference => CompositionOp::Difference, BlendingStyle::Exclusion => CompositionOp::Exclusion, BlendingStyle::Hue => CompositionOp::Hue, BlendingStyle::Saturation => CompositionOp::Saturation, BlendingStyle::Color => CompositionOp::Color, BlendingStyle::Luminosity => CompositionOp::Luminosity, } } pub fn from_str(string: &str) -> Option { match string { "multiply" => Some(BlendingStyle::Multiply), "screen" => Some(BlendingStyle::Screen), "overlay" => Some(BlendingStyle::Overlay), "darken" => Some(BlendingStyle::Darken), "lighten" => Some(BlendingStyle::Lighten), "color-dodge" => Some(BlendingStyle::ColorDodge), "color-burn" => Some(BlendingStyle::ColorBurn), "hard-light" => Some(BlendingStyle::HardLight), "soft-light" => Some(BlendingStyle::SoftLight), "difference" => Some(BlendingStyle::Difference), "exclusion" => Some(BlendingStyle::Exclusion), "hue" => Some(BlendingStyle::Hue), "saturation" => Some(BlendingStyle::Saturation), "color" => Some(BlendingStyle::Color), "luminosity" => Some(BlendingStyle::Luminosity), _ => None } } pub fn to_str(&self) -> &str { match *self { BlendingStyle::Multiply => "multiply", BlendingStyle::Screen => "screen", BlendingStyle::Overlay => "overlay", BlendingStyle::Darken => "darken", BlendingStyle::Lighten => "lighten", BlendingStyle::ColorDodge => "color-dodge", BlendingStyle::ColorBurn => "color-burn", BlendingStyle::HardLight => "hard-light", BlendingStyle::SoftLight => "soft-light", BlendingStyle::Difference => "difference", BlendingStyle::Exclusion => "exclusion", BlendingStyle::Hue => "hue", BlendingStyle::Saturation => "saturation", BlendingStyle::Color => "color", BlendingStyle::Luminosity => "luminosity", } } } #[derive(Copy, Clone, PartialEq)] pub enum CompositionOrBlending { Composition(CompositionStyle), Blending(BlendingStyle), } impl CompositionOrBlending { pub fn to_azure_style(&self) -> CompositionOp { match *self { CompositionOrBlending::Composition(op) => op.to_azure_style(), CompositionOrBlending::Blending(op) => op.to_azure_style(), } } pub fn default() -> CompositionOrBlending { CompositionOrBlending::Composition(CompositionStyle::SrcOver) } pub fn from_str(string: &str) -> Option { if let Some(op) = CompositionStyle::from_str(string) { return Some(CompositionOrBlending::Composition(op)); } if let Some(op) = BlendingStyle::from_str(string) { return Some(CompositionOrBlending::Blending(op)); } None } }