diff --git a/README.md b/README.md index 0b5483358..68b3c3cfe 100644 --- a/README.md +++ b/README.md @@ -9,7 +9,7 @@ A Rust library that makes linear color calculations and conversion easy and acce [Released](https://docs.rs/palette/0.4.1/palette/) -[Master branch](https://ogeon.github.io/docs/palette/master/palette/index.html). +[Master branch](https://ogeon.github.io/docs/palette/master/palette/index.html) ## Cargo.toml Entries @@ -31,6 +31,7 @@ These features are enabled by default: These features are disabled by default: * `"serializing"` - Enables color serializing and deserializing using `serde`. +* `"libm"` - Makes it use the `libm` floating point math library. It's only for when the `"std"` feature is disabled. ### Without the standard library @@ -40,6 +41,7 @@ Here is an example `Cargo.toml` entry for using palette on `#![no_std]`: [dependencies.palette] version = "0.4" default-features = false +features = ["libm"] # Makes it use libm instead of std for float math ``` ## It's Never "Just RGB" diff --git a/no_std_test/Cargo.toml b/no_std_test/Cargo.toml index df768b6ed..470ab20cc 100644 --- a/no_std_test/Cargo.toml +++ b/no_std_test/Cargo.toml @@ -14,4 +14,4 @@ default-features = false [dependencies.palette] path = "../palette" default-features = false -features = ["named"] +features = ["libm", "named"] diff --git a/palette/Cargo.toml b/palette/Cargo.toml index 7d8e37ee4..ee5611087 100644 --- a/palette/Cargo.toml +++ b/palette/Cargo.toml @@ -16,17 +16,20 @@ build = "build/main.rs" default = ["named_from_str", "std"] named_from_str = ["named", "phf", "phf_codegen", "std"] named = [] -std = ["approx/std", "num-traits/std"] serializing = ["serde", "std"] -#internal -strict = [] +#ignore in feature test +std = ["approx/std", "num-traits/std"] +strict = [] # Only for CI internal testing [dependencies] palette_derive = {version = "0.4.1", path = "../palette_derive"} num-traits = {version = "0.2", default-features = false} approx = {version = "0.3", default-features = false} -libm = "0.1" + +[dependencies.libm] +version = "0.1" +optional = true [dependencies.phf] version = "0.7" diff --git a/palette/README.md b/palette/README.md deleted file mode 100644 index 32bb56dd2..000000000 --- a/palette/README.md +++ /dev/null @@ -1,152 +0,0 @@ -# palette - -[![Build Status](https://travis-ci.org/Ogeon/palette.svg?branch=master)](https://travis-ci.org/Ogeon/palette) -[![Crates.io](https://img.shields.io/crates/v/palette.svg)](https://crates.io/crates/palette/) - -A Rust library that makes linear color calculations and conversion easy and accessible for anyone. It provides both precision tools that lets you work in exactly the color space you want to, as well as a general color type that abstracts away some of the technical details. - -## Online Documentation - -[Released](https://docs.rs/palette/0.4.0/palette/) - -[Master branch](https://ogeon.github.io/docs/palette/master/palette/index.html). - -## Cargo.toml Entries - -Add the following lines to your `Cargo.toml` file: - -```toml -[dependencies] -palette = "0.4" -``` - -### Features - -These features are enabled by default: - -* `"named"` - Enables color constants, located in the `named` module. -* `"named_from_str"` - Enables the `named::from_str`, which maps name string to colors. This requires the standard library. -* `"std"` - Enables use of the standard library. - -These features are disabled by default: - -* `"serializing"` - Enables color serializing and deserializing using `serde`. - -### Without the standard library - -Here is an example `Cargo.toml` entry for using palette on `#![no_std]`: - -```toml -[dependencies.palette] -version = "0.4" -default-features = false -``` - -## It's Never "Just RGB" - -Colors in, for example, images are often "gamma corrected" or stored in sRGB format as a compression method and to prevent banding. This is also a bit of a legacy from the ages of the CRT monitors, where the output from the electron gun was nonlinear. The problem is that these formats doesn't represent the actual intensities, and the compression has to be reverted to make sure that any operations on the colors are accurate. This library uses a completely linear work flow, and comes with the tools for transitioning between linear and non-linear RGB. - -Adding to that, there are more than one kind of non-linear RGB. Ironically enough, this turns RGB into one of the most complex color spaces. - -## What It Can Do - -Palette provides tools for both color manipulation and conversion between color spaces. These are some highlights. - -### Color Spaces - -RGB is probably the most widely known color space, but it's not the only one. You have probably used a color picker with a rainbow wheel and a brightness slider. That may have been an HSV or an HSL color picker, where the color is encoded as hue, saturation and brightness/lightness. There's also a group of color spaces that are designed to be perceptually uniform, meaning that the perceptual change is equal to the numerical change. - -Selecting the proper color space can have a big impact on how the resulting image looks (as illustrated by some of the programs in `examples`), and Palette makes the conversion between them as easy as a call to `from` or `into`. - -This example takes an sRGB color, converts it to CIE L\*C\*h°, shifts its hue by 180° and converts it back to RGB: - -```Rust -extern crate palette; -use palette::{Srgb, LinSrgb, Lch, Hue}; - -let lch_color: Lch = Srgb::new(0.8, 0.2, 0.1).into(); -let new_color = LinSrgb::from(lch_color.shift_hue(180.0)); -``` - -This results in the following two colors: - -![Hue Shift Comparison](gfx/readme_color_spaces.png) - -### Manipulation - -Palette comes with a number of color manipulation tools, that are implemented as traits. These includes lighten/darken, saturate/desaturate and hue shift. These traits are only implemented on types where they are meaningful, which means that you can't shift the hue of an RGB color without converting it to a color space where it makes sense. - -This may seem limiting, but the point is to avoid inconsistent behavior due to arbitrary defaults, such as saturating a gray color to red when there is no available hue information. The abstract `Color` type does still support every operation, for when this is less important. - -The following example shows how the `Color` type is used to make a lighter and a desaturated version of the original. - -```Rust -extern crate palette; -use palette::{Color, Srgb, Shade, Saturate}; - -let color: Color = Srgb::new(0.8, 0.2, 0.1).into_linear().into(); -let lighter = color.lighten(0.1); -let desaturated = color.desaturate(0.5); -``` - -This results in the following three colors: - -![Manipulation Comparison](gfx/readme_manipulation.png) - -### Gradients - -There is also a linear gradient type which makes it easy to interpolate between a series of colors. This gradient can be used in any color space and it can be used to make color sequence iterators. - -The following example shows two gradients between the same two endpoints, but one is in RGB and the other in is HSV space. - -```Rust -extern crate palette; -use palette::{LinSrgb, Hsv, Gradient}; - -let grad1 = Gradient::new(vec![ - LinSrgb::new(1.0, 0.1, 0.1), - LinSrgb::new(0.1, 1.0, 1.0) -]); - -let grad2 = Gradient::new(vec![ - Hsv::from(LinSrgb::new(1.0, 0.1, 0.1)), - Hsv::from(LinSrgb::new(0.1, 1.0, 1.0)) -]); -``` - -The RGB gradient goes through gray, while the HSV gradients changes only the hue: - -![Gradient Comparison](gfx/readme_gradients.png) - -## What It Isn't - -This library is only meant for color manipulation and conversion. It's not a fully features image manipulation library. It will only handle colors, and not whole images. There are features that are meant to work as bridges between Palette and other graphical libraries, but the main features are limited to only focus on single pixel operations, to keep the scope at a manageable size. - -[pixel_module]: https://ogeon.github.io/docs/palette/master/palette/pixel/index.html - -## Using palette in an embedded environment - -Palette supports `#![no_std]` environments by disabling the `"std"` feature. However, there are some things that are unavailable without the standard library: - -* Gradients are unavailable, because they depend heavily on Vectors -* The `"named_from_str"` feature requires the standard library as well -* Serialization using `serde` is unavailable - -It uses [`libm`] to provide the floating-point operations that are typically in `std`. - -[`libm`]: /~https://github.com/japaric/libm - -## Contributing - -All sorts of contributions are welcome, no matter how huge or tiny, so take a look at [CONTRIBUTING.md](CONTRIBUTING.md) for guidelines, if you are interested. - -## License - -Licensed under either of - -* Apache License, Version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or ) -* MIT license ([LICENSE-MIT](LICENSE-MIT) or ) - -at your option. - -Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions. diff --git a/palette/README.md b/palette/README.md new file mode 120000 index 000000000..32d46ee88 --- /dev/null +++ b/palette/README.md @@ -0,0 +1 @@ +../README.md \ No newline at end of file diff --git a/palette/src/float.rs b/palette/src/float.rs index 6976b9565..02d36dcb0 100644 --- a/palette/src/float.rs +++ b/palette/src/float.rs @@ -17,7 +17,9 @@ pub use self::no_std_float_trait::Float; #[cfg(not(feature = "std"))] mod no_std_float_trait { + #[cfg(feature = "libm")] extern crate libm; + #[cfg(feature = "libm")] use self::libm::{F32Ext, F64Ext}; use core::{f32, f64}; @@ -61,6 +63,7 @@ mod no_std_float_trait { fn round(self) -> Self; } + #[cfg(feature = "libm")] impl Float for f32 { fn sqrt(self) -> f32 { F32Ext::cbrt(self) @@ -85,6 +88,7 @@ mod no_std_float_trait { } } + #[cfg(feature = "libm")] impl Float for f64 { fn sqrt(self) -> f64 { F64Ext::sqrt(self) @@ -108,4 +112,57 @@ mod no_std_float_trait { F64Ext::round(self) } } + + #[cfg(not(feature = "libm"))] + impl Float for f32 { + fn sqrt(self) -> f32 { + panic!("need to select a float library for palette") + } + fn cbrt(self) -> f32 { + panic!("need to select a float library for palette") + } + fn powf(self, other: f32) -> f32 { + panic!("need to select a float library for palette") + } + fn sin(self) -> f32 { + panic!("need to select a float library for palette") + } + fn cos(self) -> f32 { + panic!("need to select a float library for palette") + } + fn atan2(self, other: f32) -> f32 { + panic!("need to select a float library for palette") + } + fn round(self) -> f32 { + panic!("need to select a float library for palette") + } + } + + #[cfg(not(feature = "libm"))] + impl Float for f64 { + fn sqrt(self) -> f64 { + panic!("need to select a float library for palette") + } + fn cbrt(self) -> f64 { + panic!("need to select a float library for palette") + } + fn powf(self, other: f64) -> f64 { + panic!("need to select a float library for palette") + } + fn sin(self) -> f64 { + panic!("need to select a float library for palette") + } + fn cos(self) -> f64 { + panic!("need to select a float library for palette") + } + fn atan2(self, other: f64) -> f64 { + panic!("need to select a float library for palette") + } + fn round(self) -> f64 { + panic!("need to select a float library for palette") + } + } } + +#[cfg(not(any(feature = "std", feature = "libm")))] +compile_error!("The palette crate needs a float library. Please enable the \"std\" or \"libm\" feature."); diff --git a/scripts/test_features.sh b/scripts/test_features.sh index 385773ac7..bf0a6ad4a 100644 --- a/scripts/test_features.sh +++ b/scripts/test_features.sh @@ -4,7 +4,7 @@ set -e features="" #Features that will always be activated -required_features="strict" +required_features="std strict" #Find features @@ -14,7 +14,7 @@ current_dependency="" while read -r line || [[ -n "$line" ]]; do if [[ "$line" == "[features]" ]]; then walking_features=true - elif [[ $walking_features == true ]] && [[ "$line" == "#internal" ]]; then + elif [[ $walking_features == true ]] && [[ "$line" == "#ignore in feature test" ]]; then walking_features=false elif [[ $walking_features == true ]] && echo "$line" | grep -E "^\[.*\]" > /dev/null; then walking_features=false