use core::fmt;
use crate::encode::{Encode, Encoding, RefEncode};
use crate::ffi;
#[repr(transparent)]
#[derive(Copy, Clone, Default)]
pub struct Bool {
value: ffi::BOOL,
}
impl Bool {
pub const YES: Self = Self::from_raw(ffi::YES);
pub const NO: Self = Self::from_raw(ffi::NO);
#[inline]
pub const fn new(value: bool) -> Self {
let value = value as ffi::BOOL;
Self { value }
}
#[inline]
pub const fn from_raw(value: ffi::BOOL) -> Self {
Self { value }
}
#[inline]
pub const fn as_raw(self) -> ffi::BOOL {
self.value
}
#[inline]
pub const fn is_false(self) -> bool {
!self.as_bool()
}
#[inline]
pub const fn is_true(self) -> bool {
self.as_bool()
}
#[inline]
pub const fn as_bool(self) -> bool {
self.value != ffi::NO
}
}
impl From<bool> for Bool {
#[inline]
fn from(b: bool) -> Bool {
Bool::new(b)
}
}
impl From<Bool> for bool {
#[inline]
fn from(b: Bool) -> bool {
b.as_bool()
}
}
impl fmt::Debug for Bool {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(if self.as_bool() { "YES" } else { "NO" })
}
}
trait Helper {
const __ENCODING: Encoding;
}
impl<T: Encode> Helper for T {
const __ENCODING: Encoding = T::ENCODING;
}
impl Helper for bool {
const __ENCODING: Encoding = Encoding::Bool;
}
unsafe impl Encode for Bool {
const ENCODING: Encoding = ffi::BOOL::__ENCODING;
}
unsafe impl RefEncode for Bool {
const ENCODING_REF: Encoding = Encoding::Pointer(&Self::ENCODING);
}
#[cfg(test)]
mod tests {
use super::*;
use crate::__macro_helpers::{ConvertArgument, ConvertReturn};
use alloc::format;
#[test]
fn test_basic() {
let b = Bool::new(true);
assert!(b.as_bool());
assert!(b.is_true());
assert!(!b.is_false());
assert!(bool::from(b));
assert_eq!(b.as_raw() as usize, 1);
let b = Bool::new(false);
assert!(!b.as_bool());
assert!(!b.is_true());
assert!(b.is_false());
assert!(!bool::from(b));
assert_eq!(b.as_raw() as usize, 0);
}
#[test]
fn test_associated_constants() {
let b = Bool::YES;
assert!(b.as_bool());
assert!(b.is_true());
assert_eq!(b.as_raw() as usize, 1);
let b = Bool::NO;
assert!(!b.as_bool());
assert!(b.is_false());
assert_eq!(b.as_raw() as usize, 0);
}
#[test]
fn test_encode() {
assert_eq!(bool::__ENCODING, Encoding::Bool);
assert_eq!(
<bool as ConvertArgument>::__Inner::__ENCODING,
<bool as ConvertArgument>::__Inner::ENCODING
);
assert_eq!(
<bool as ConvertReturn>::__Inner::__ENCODING,
<bool as ConvertReturn>::__Inner::ENCODING
);
}
#[test]
fn test_impls() {
let b: Bool = Default::default();
assert!(!b.as_bool());
assert!(b.is_false());
assert!(Bool::from(true).as_bool());
assert!(Bool::from(true).is_true());
assert!(Bool::from(false).is_false());
assert!(Bool::from(true).is_true());
assert!(Bool::from(false).is_false());
}
#[test]
fn test_debug() {
assert_eq!(format!("{:?}", Bool::from(true)), "YES");
assert_eq!(format!("{:?}", Bool::from(false)), "NO");
}
#[test]
#[cfg(all(target_vendor = "apple", target_os = "macos", target_arch = "x86_64"))]
fn test_outside_normal() {
let b = Bool::from_raw(42);
assert!(b.is_true());
assert!(!b.is_false());
assert_eq!(b.as_raw(), 42);
}
}