add docs for blend

This commit is contained in:
Corwin 2022-08-06 21:15:38 +01:00
parent 921c26f7c2
commit afa9ef9109

View file

@ -1,21 +1,48 @@
//! This controls the blending modes on the GBA.
//!
//! For now a description of how blending can be used is found on [the tonc page
//! for graphic
//! effects](https://www.coranac.com/tonc/text/gfx.htm#ssec-bld-gba). See the
//! [Blend] struct for all the functions to manage blend effects. You accquire
//! the Blend struct through the [Display][super::Display] struct.
//! ```no_run
//! # #![no_main]
//! # #![no_std]
//! # fn blend(mut gba: agb::Gba) {
//! let mut blend = gba.display.blend.get();
//! // ...
//! # }
//! ```
//! where `gba` is a mutable [Gba][crate::Gba] struct.
use crate::{fixnum::Num, memory_mapped::set_bits}; use crate::{fixnum::Num, memory_mapped::set_bits};
use super::tiled::BackgroundID; use super::tiled::BackgroundID;
/// The layers, top layer will be blended into the bottom layer
#[derive(Clone, Copy, Debug)] #[derive(Clone, Copy, Debug)]
pub enum Layer { pub enum Layer {
/// Top layer gets blended into the bottom layer
Top = 0, Top = 0,
/// The bottom layer of the blend
Bottom = 1, Bottom = 1,
} }
/// The different blend modes avaliable on the GBA
#[derive(Clone, Copy, Debug)] #[derive(Clone, Copy, Debug)]
pub enum BlendMode { pub enum BlendMode {
// No blending
Off = 0, Off = 0,
// Aditive blending, use the [Blend::set_blend_weight] function to use this
Normal = 0b01, Normal = 0b01,
// Brighten, use the [Blend::set_fade] to use this
FadeToWhite = 0b10, FadeToWhite = 0b10,
// Darken, use the [Blend::set_fade] to use this
FadeToBlack = 0b11, FadeToBlack = 0b11,
} }
/// Manages the blending, won't cause anything to change unless [Blend::commit]
/// is called.
pub struct Blend { pub struct Blend {
targets: u16, targets: u16,
blend_weights: u16, blend_weights: u16,
@ -39,28 +66,34 @@ impl Blend {
blend blend
} }
/// Reset the targets to all disabled, the targets control which layers are
/// enabled for blending.
pub fn reset_targets(&mut self) -> &mut Self { pub fn reset_targets(&mut self) -> &mut Self {
self.targets = 0; self.targets = 0;
self self
} }
/// Reset the blend weights
pub fn reset_weights(&mut self) -> &mut Self { pub fn reset_weights(&mut self) -> &mut Self {
self.blend_weights = 0; self.blend_weights = 0;
self self
} }
/// Reset the brighten and darken weights
pub fn reset_fades(&mut self) -> &mut Self { pub fn reset_fades(&mut self) -> &mut Self {
self.fade_weight = 0; self.fade_weight = 0;
self self
} }
/// Reset targers, blend weights, and fades
pub fn reset(&mut self) -> &mut Self { pub fn reset(&mut self) -> &mut Self {
self.reset_targets().reset_fades().reset_weights() self.reset_targets().reset_fades().reset_weights()
} }
/// Set whether a background is enabled for blending on a particular layer.
pub fn set_background_enable( pub fn set_background_enable(
&mut self, &mut self,
layer: Layer, layer: Layer,
@ -73,6 +106,7 @@ impl Blend {
self self
} }
/// Set whether objects are enabled for blending on a particular layer
pub fn set_object_enable(&mut self, layer: Layer, enable: bool) -> &mut Self { pub fn set_object_enable(&mut self, layer: Layer, enable: bool) -> &mut Self {
let bit_to_modify = 0x4 + (layer as usize * 8); let bit_to_modify = 0x4 + (layer as usize * 8);
self.targets = set_bits(self.targets, enable as u16, 1, bit_to_modify); self.targets = set_bits(self.targets, enable as u16, 1, bit_to_modify);
@ -80,6 +114,9 @@ impl Blend {
self self
} }
/// Set whether the backdrop contributes to the blend on a particular layer.
/// The backdrop is transparent colour, the colour rendered when nothing is
/// in it's place.
pub fn set_backdrop_enable(&mut self, layer: Layer, enable: bool) -> &mut Self { pub fn set_backdrop_enable(&mut self, layer: Layer, enable: bool) -> &mut Self {
let bit_to_modify = 0x5 + (layer as usize * 8); let bit_to_modify = 0x5 + (layer as usize * 8);
self.targets = set_bits(self.targets, enable as u16, 1, bit_to_modify); self.targets = set_bits(self.targets, enable as u16, 1, bit_to_modify);
@ -87,6 +124,7 @@ impl Blend {
self self
} }
/// Set the weight for the blend on a particular layer.
pub fn set_blend_weight(&mut self, layer: Layer, value: Num<u8, 4>) -> &mut Self { pub fn set_blend_weight(&mut self, layer: Layer, value: Num<u8, 4>) -> &mut Self {
self.blend_weights = set_bits( self.blend_weights = set_bits(
self.blend_weights, self.blend_weights,
@ -98,18 +136,22 @@ impl Blend {
self self
} }
/// Set the fade of brighten or darken
pub fn set_fade(&mut self, value: Num<u8, 4>) -> &mut Self { pub fn set_fade(&mut self, value: Num<u8, 4>) -> &mut Self {
self.fade_weight = value.to_raw() as u16; self.fade_weight = value.to_raw() as u16;
self self
} }
/// Set the current blend mode
pub fn set_blend_mode(&mut self, blend_mode: BlendMode) -> &mut Self { pub fn set_blend_mode(&mut self, blend_mode: BlendMode) -> &mut Self {
self.targets = set_bits(self.targets, blend_mode as u16, 2, 0x6); self.targets = set_bits(self.targets, blend_mode as u16, 2, 0x6);
self self
} }
/// Commits the current state, should be called near after a call to wait
/// for next vblank.
pub fn commit(&self) { pub fn commit(&self) {
unsafe { unsafe {
BLEND_CONTROL.write_volatile(self.targets); BLEND_CONTROL.write_volatile(self.targets);