extern crate rand;
use self::rand::Rng;
use self::rand::distributions::{Distribution, Standard};
use libc::uint32_t;
use num::FromPrimitive;
use std::mem::transmute;
use crate::sys;
use crate::get_error;
pub struct Palette {
raw: *mut sys::SDL_Palette
}
impl Palette {
#[inline]
pub fn new(mut capacity: usize) -> Result<Self, String> {
use crate::common::*;
let ncolors = {
let u32_max = u32::max_value() as usize;
if capacity > u32_max { capacity = u32_max; }
match validate_int(capacity as u32, "capacity") {
Ok(len) => len,
Err(e) => return Err(format!("{}", e)),
}
};
let raw = unsafe { sys::SDL_AllocPalette(ncolors) };
if raw.is_null() {
Err(get_error())
} else {
Ok(Palette {
raw: raw,
})
}
}
pub fn with_colors(colors: &[Color]) -> Result<Self, String> {
let pal = r#try!(Self::new(colors.len()));
let ncolors = colors.len() as ::libc::c_int;
let result = unsafe {
let mut raw_colors: Vec<sys::SDL_Color> = colors.iter()
.map(|color| color.raw())
.collect();
let pal_ptr = (&mut raw_colors[0]) as *mut sys::SDL_Color;
sys::SDL_SetPaletteColors(pal.raw, pal_ptr, 0, ncolors)
};
if result < 0 {
Err(get_error())
} else {
Ok(pal)
}
}
pub fn len(&self) -> usize {
unsafe { (*self.raw).ncolors as usize }
}
pub fn is_empty(&self) -> bool {
self.len() == 0
}
}
impl Drop for Palette {
fn drop(&mut self) {
unsafe { sys::SDL_FreePalette(self.raw); }
}
}
impl_raw_accessors!((Palette, *mut sys::SDL_Palette));
#[test]
fn create_palette() {
let colors: Vec<_> = (0 .. 0xff).map(|u| {
Color::RGB(u, 0, 0xff - u)
}).collect();
let palette = Palette::with_colors(&colors).unwrap();
assert!(palette.len() == 255);
}
#[derive(Copy, Clone, Eq, PartialEq, Hash, Debug)]
pub struct Color {
pub r: u8,
pub g: u8,
pub b: u8,
pub a: u8
}
impl Color {
#[inline]
#[allow(non_snake_case)]
pub fn RGB(r: u8, g: u8, b: u8) -> Color {
Color { r: r, g: g, b: b, a: 0xff }
}
#[inline]
#[allow(non_snake_case)]
pub fn RGBA(r: u8, g: u8, b: u8, a: u8) -> Color {
Color { r: r, g: g, b: b, a: a }
}
pub fn to_u32(&self, format: &PixelFormat) -> u32 {
unsafe { sys::SDL_MapRGBA(format.raw, self.r, self.g, self.b, self.a) }
}
pub fn from_u32(format: &PixelFormat, pixel: u32) -> Color {
let (mut r, mut g, mut b, mut a) = (0, 0, 0, 0);
unsafe {
sys::SDL_GetRGBA(pixel, format.raw, &mut r, &mut g, &mut b, &mut a)
};
Color::RGBA(r, g, b, a)
}
#[inline]
pub fn rgb(&self) -> (u8, u8, u8) {
(self.r, self.g, self.b)
}
#[inline]
pub fn rgba(&self) -> (u8, u8, u8, u8) {
(self.r, self.g, self.b, self.a)
}
#[inline]
fn raw(&self) -> sys::SDL_Color {
sys::SDL_Color { r: self.r, g: self.g, b: self.b, a: self.a }
}
}
impl Into<sys::SDL_Color> for Color {
fn into(self) -> sys::SDL_Color {
self.raw()
}
}
impl From<sys::SDL_Color> for Color {
fn from(raw: sys::SDL_Color) -> Color {
Color::RGBA(raw.r, raw.g, raw.b, raw.a)
}
}
impl From<(u8, u8, u8)> for Color {
fn from((r, g, b): (u8, u8, u8)) -> Color {
Color::RGB(r, g, b)
}
}
impl From<(u8, u8, u8, u8)> for Color {
fn from((r, g, b, a): (u8, u8, u8, u8)) -> Color {
Color::RGBA(r, g, b, a)
}
}
impl Distribution<Color> for Standard {
fn sample<R: Rng + ?Sized>(&self, rng: &mut R) -> Color {
if rng.gen() {
Color::RGBA(rng.gen(), rng.gen(), rng.gen(), rng.gen())
} else {
Color::RGB(rng.gen(), rng.gen(), rng.gen())
}
}
}
pub struct PixelMasks {
pub bpp: u8,
pub rmask: u32,
pub gmask: u32,
pub bmask: u32,
pub amask: u32
}
pub struct PixelFormat {
raw: *mut sys::SDL_PixelFormat
}
impl_raw_accessors!((PixelFormat, *mut sys::SDL_PixelFormat));
impl_raw_constructor!((PixelFormat, PixelFormat (raw: *mut sys::SDL_PixelFormat)));
#[repr(i32)]
#[derive(Copy, Clone, Eq, PartialEq, Hash, Debug)]
pub enum PixelFormatEnum {
Unknown = sys::SDL_PIXELFORMAT_UNKNOWN as i32,
Index1LSB = sys::SDL_PIXELFORMAT_INDEX1LSB as i32,
Index1MSB = sys::SDL_PIXELFORMAT_INDEX1MSB as i32,
Index4LSB = sys::SDL_PIXELFORMAT_INDEX4LSB as i32,
Index4MSB = sys::SDL_PIXELFORMAT_INDEX4MSB as i32,
Index8 = sys::SDL_PIXELFORMAT_INDEX8 as i32,
RGB332 = sys::SDL_PIXELFORMAT_RGB332 as i32,
RGB444 = sys::SDL_PIXELFORMAT_RGB444 as i32,
RGB555 = sys::SDL_PIXELFORMAT_RGB555 as i32,
BGR555 = sys::SDL_PIXELFORMAT_BGR555 as i32,
ARGB4444 = sys::SDL_PIXELFORMAT_ARGB4444 as i32,
RGBA4444 = sys::SDL_PIXELFORMAT_RGBA4444 as i32,
ABGR4444 = sys::SDL_PIXELFORMAT_ABGR4444 as i32,
BGRA4444 = sys::SDL_PIXELFORMAT_BGRA4444 as i32,
ARGB1555 = sys::SDL_PIXELFORMAT_ARGB1555 as i32,
RGBA5551 = sys::SDL_PIXELFORMAT_RGBA5551 as i32,
ABGR1555 = sys::SDL_PIXELFORMAT_ABGR1555 as i32,
BGRA5551 = sys::SDL_PIXELFORMAT_BGRA5551 as i32,
RGB565 = sys::SDL_PIXELFORMAT_RGB565 as i32,
BGR565 = sys::SDL_PIXELFORMAT_BGR565 as i32,
RGB24 = sys::SDL_PIXELFORMAT_RGB24 as i32,
BGR24 = sys::SDL_PIXELFORMAT_BGR24 as i32,
RGB888 = sys::SDL_PIXELFORMAT_RGB888 as i32,
RGBX8888 = sys::SDL_PIXELFORMAT_RGBX8888 as i32,
BGR888 = sys::SDL_PIXELFORMAT_BGR888 as i32,
BGRX8888 = sys::SDL_PIXELFORMAT_BGRX8888 as i32,
ARGB8888 = sys::SDL_PIXELFORMAT_ARGB8888 as i32,
RGBA8888 = sys::SDL_PIXELFORMAT_RGBA8888 as i32,
ABGR8888 = sys::SDL_PIXELFORMAT_ABGR8888 as i32,
BGRA8888 = sys::SDL_PIXELFORMAT_BGRA8888 as i32,
ARGB2101010 = sys::SDL_PIXELFORMAT_ARGB2101010 as i32,
YV12 = sys::SDL_PIXELFORMAT_YV12 as i32,
IYUV = sys::SDL_PIXELFORMAT_IYUV as i32,
YUY2 = sys::SDL_PIXELFORMAT_YUY2 as i32,
UYVY = sys::SDL_PIXELFORMAT_UYVY as i32,
YVYU = sys::SDL_PIXELFORMAT_YVYU as i32
}
#[cfg(target_endian = "big")]
impl PixelFormatEnum {
pub const RGBA32: PixelFormatEnum = PixelFormatEnum::RGBA8888;
pub const ARGB32: PixelFormatEnum = PixelFormatEnum::ARGB8888;
pub const BGRA32: PixelFormatEnum = PixelFormatEnum::BGRA8888;
pub const ABGR32: PixelFormatEnum = PixelFormatEnum::ABGR8888;
}
#[cfg(target_endian = "little")]
impl PixelFormatEnum {
pub const RGBA32: PixelFormatEnum = PixelFormatEnum::ABGR8888;
pub const ARGB32: PixelFormatEnum = PixelFormatEnum::BGRA8888;
pub const BGRA32: PixelFormatEnum = PixelFormatEnum::ARGB8888;
pub const ABGR32: PixelFormatEnum = PixelFormatEnum::RGBA8888;
}
impl PixelFormatEnum {
pub fn from_masks(masks: PixelMasks) -> PixelFormatEnum {
unsafe {
let format = sys::SDL_MasksToPixelFormatEnum(masks.bpp as i32, masks.rmask, masks.gmask, masks.bmask, masks.amask);
PixelFormatEnum::from_u64(format as u64).unwrap()
}
}
pub fn into_masks(self) -> Result<PixelMasks, String> {
let format: u32 = self as u32;
let mut bpp = 0;
let mut rmask = 0;
let mut gmask = 0;
let mut bmask = 0;
let mut amask = 0;
let result = unsafe {
sys::SDL_PixelFormatEnumToMasks(format, &mut bpp, &mut rmask, &mut gmask, &mut bmask, &mut amask)
};
if result == sys::SDL_bool::SDL_FALSE {
Err(get_error())
} else {
Ok(PixelMasks {
bpp: bpp as u8,
rmask: rmask,
gmask: gmask,
bmask: bmask,
amask: amask
})
}
}
pub fn byte_size_from_pitch_and_height(&self, pitch: usize, height: usize) -> usize {
match *self {
PixelFormatEnum::YV12 | PixelFormatEnum::IYUV => {
pitch * height + 2 * (pitch / 2 * height / 2)
},
_ => pitch * height
}
}
#[cfg_attr(feature = "cargo-clippy", allow(match_same_arms))]
pub fn byte_size_of_pixels(&self, num_of_pixels: usize) -> usize {
match *self {
PixelFormatEnum::RGB332
=> num_of_pixels,
PixelFormatEnum::RGB444 | PixelFormatEnum::RGB555 |
PixelFormatEnum::BGR555 | PixelFormatEnum::ARGB4444 |
PixelFormatEnum::RGBA4444 | PixelFormatEnum::ABGR4444 |
PixelFormatEnum::BGRA4444 | PixelFormatEnum::ARGB1555 |
PixelFormatEnum::RGBA5551 | PixelFormatEnum::ABGR1555 |
PixelFormatEnum::BGRA5551 | PixelFormatEnum::RGB565 |
PixelFormatEnum::BGR565
=> num_of_pixels * 2,
PixelFormatEnum::RGB24 | PixelFormatEnum::BGR24
=> num_of_pixels * 3,
PixelFormatEnum::RGB888 | PixelFormatEnum::RGBX8888 |
PixelFormatEnum::BGR888 | PixelFormatEnum::BGRX8888 |
PixelFormatEnum::ARGB8888 | PixelFormatEnum::RGBA8888 |
PixelFormatEnum::ABGR8888 | PixelFormatEnum::BGRA8888 |
PixelFormatEnum::ARGB2101010
=> num_of_pixels * 4,
PixelFormatEnum::YV12 | PixelFormatEnum::IYUV
=> num_of_pixels / 2 * 3,
PixelFormatEnum::YUY2 | PixelFormatEnum::UYVY |
PixelFormatEnum::YVYU
=> num_of_pixels * 2,
PixelFormatEnum::Index8
=> num_of_pixels,
PixelFormatEnum::Unknown | PixelFormatEnum::Index1LSB |
PixelFormatEnum::Index1MSB | PixelFormatEnum::Index4LSB |
PixelFormatEnum::Index4MSB
=> panic!("not supported format: {:?}", *self),
}
}
#[cfg_attr(feature = "cargo-clippy", allow(match_same_arms))]
pub fn byte_size_per_pixel(&self) -> usize {
match *self {
PixelFormatEnum::RGB332
=> 1,
PixelFormatEnum::RGB444 | PixelFormatEnum::RGB555 |
PixelFormatEnum::BGR555 | PixelFormatEnum::ARGB4444 |
PixelFormatEnum::RGBA4444 | PixelFormatEnum::ABGR4444 |
PixelFormatEnum::BGRA4444 | PixelFormatEnum::ARGB1555 |
PixelFormatEnum::RGBA5551 | PixelFormatEnum::ABGR1555 |
PixelFormatEnum::BGRA5551 | PixelFormatEnum::RGB565 |
PixelFormatEnum::BGR565
=> 2,
PixelFormatEnum::RGB24 | PixelFormatEnum::BGR24
=> 3,
PixelFormatEnum::RGB888 | PixelFormatEnum::RGBX8888 |
PixelFormatEnum::BGR888 | PixelFormatEnum::BGRX8888 |
PixelFormatEnum::ARGB8888 | PixelFormatEnum::RGBA8888 |
PixelFormatEnum::ABGR8888 | PixelFormatEnum::BGRA8888 |
PixelFormatEnum::ARGB2101010
=> 4,
PixelFormatEnum::YV12 | PixelFormatEnum::IYUV
=> 2,
PixelFormatEnum::YUY2 | PixelFormatEnum::UYVY |
PixelFormatEnum::YVYU
=> 2,
PixelFormatEnum::Index8
=> 1,
PixelFormatEnum::Unknown | PixelFormatEnum::Index1LSB |
PixelFormatEnum::Index1MSB | PixelFormatEnum::Index4LSB |
PixelFormatEnum::Index4MSB
=> panic!("not supported format: {:?}", *self),
}
}
pub fn supports_alpha(&self) -> bool {
use crate::pixels::PixelFormatEnum::*;
match *self {
ARGB4444 | ARGB1555 | ARGB8888 | ARGB2101010 |
ABGR4444 | ABGR1555 | ABGR8888 |
BGRA4444 | BGRA5551 | BGRA8888 |
RGBA4444 | RGBA5551 | RGBA8888 => true,
_ => false
}
}
}
impl Into<uint32_t> for PixelFormatEnum {
fn into(self) -> uint32_t {
self as uint32_t
}
}
impl From<PixelFormat> for PixelFormatEnum {
fn from(pf: PixelFormat) -> PixelFormatEnum {
unsafe {
let ref sdl_pf = *pf.raw;
match PixelFormatEnum::from_u64(sdl_pf.format as u64) {
Some(pfe) => pfe,
None => panic!("Unknown pixel format: {:?}", sdl_pf.format)
}
}
}
}
impl FromPrimitive for PixelFormatEnum {
fn from_i64(n: i64) -> Option<PixelFormatEnum> {
use self::PixelFormatEnum::*;
let n = n as u32;
Some( match unsafe { transmute(n) } {
sys::SDL_PIXELFORMAT_UNKNOWN => Unknown,
sys::SDL_PIXELFORMAT_INDEX1LSB => Index1LSB,
sys::SDL_PIXELFORMAT_INDEX1MSB => Index1MSB,
sys::SDL_PIXELFORMAT_INDEX4LSB => Index4LSB,
sys::SDL_PIXELFORMAT_INDEX4MSB => Index4MSB,
sys::SDL_PIXELFORMAT_INDEX8 => Index8,
sys::SDL_PIXELFORMAT_RGB332 => RGB332,
sys::SDL_PIXELFORMAT_RGB444 => RGB444,
sys::SDL_PIXELFORMAT_RGB555 => RGB555,
sys::SDL_PIXELFORMAT_BGR555 => BGR555,
sys::SDL_PIXELFORMAT_ARGB4444 => ARGB4444,
sys::SDL_PIXELFORMAT_RGBA4444 => RGBA4444,
sys::SDL_PIXELFORMAT_ABGR4444 => ABGR4444,
sys::SDL_PIXELFORMAT_BGRA4444 => BGRA4444,
sys::SDL_PIXELFORMAT_ARGB1555 => ARGB1555,
sys::SDL_PIXELFORMAT_RGBA5551 => RGBA5551,
sys::SDL_PIXELFORMAT_ABGR1555 => ABGR1555,
sys::SDL_PIXELFORMAT_BGRA5551 => BGRA5551,
sys::SDL_PIXELFORMAT_RGB565 => RGB565,
sys::SDL_PIXELFORMAT_BGR565 => BGR565,
sys::SDL_PIXELFORMAT_RGB24 => RGB24,
sys::SDL_PIXELFORMAT_BGR24 => BGR24,
sys::SDL_PIXELFORMAT_RGB888 => RGB888,
sys::SDL_PIXELFORMAT_RGBX8888 => RGBX8888,
sys::SDL_PIXELFORMAT_BGR888 => BGR888,
sys::SDL_PIXELFORMAT_BGRX8888 => BGRX8888,
sys::SDL_PIXELFORMAT_ARGB8888 => ARGB8888,
sys::SDL_PIXELFORMAT_RGBA8888 => RGBA8888,
sys::SDL_PIXELFORMAT_ABGR8888 => ABGR8888,
sys::SDL_PIXELFORMAT_BGRA8888 => BGRA8888,
sys::SDL_PIXELFORMAT_ARGB2101010 => ARGB2101010,
sys::SDL_PIXELFORMAT_YV12 => YV12,
sys::SDL_PIXELFORMAT_IYUV => IYUV,
sys::SDL_PIXELFORMAT_YUY2 => YUY2,
sys::SDL_PIXELFORMAT_UYVY => UYVY,
sys::SDL_PIXELFORMAT_YVYU => YVYU,
_ => return None,
})
}
fn from_u64(n: u64) -> Option<PixelFormatEnum> { FromPrimitive::from_i64(n as i64) }
}
#[test]
fn test_pixel_format_enum() {
let pixel_formats = vec![
PixelFormatEnum::RGB332,
PixelFormatEnum::RGB444, PixelFormatEnum::RGB555,
PixelFormatEnum::BGR555, PixelFormatEnum::ARGB4444,
PixelFormatEnum::RGBA4444, PixelFormatEnum::ABGR4444,
PixelFormatEnum::BGRA4444, PixelFormatEnum::ARGB1555,
PixelFormatEnum::RGBA5551, PixelFormatEnum::ABGR1555,
PixelFormatEnum::BGRA5551, PixelFormatEnum::RGB565,
PixelFormatEnum::BGR565,
PixelFormatEnum::RGB24, PixelFormatEnum::BGR24,
PixelFormatEnum::RGB888, PixelFormatEnum::RGBX8888,
PixelFormatEnum::BGR888, PixelFormatEnum::BGRX8888,
PixelFormatEnum::ARGB8888, PixelFormatEnum::RGBA8888,
PixelFormatEnum::ABGR8888, PixelFormatEnum::BGRA8888,
PixelFormatEnum::ARGB2101010,
PixelFormatEnum::YV12, PixelFormatEnum::IYUV,
PixelFormatEnum::YUY2, PixelFormatEnum::UYVY,
PixelFormatEnum::YVYU,
PixelFormatEnum::Index8,
];
let _sdl_context = crate::sdl::init().unwrap();
for format in pixel_formats {
if let Ok(surf) = super::surface::Surface::new(1, 1, format) {
let surf_format = surf.pixel_format();
assert_eq!(PixelFormatEnum::from(surf_format), format);
}
}
}