use std::ops::{Add, AddAssign, Div};
use allocative::Allocative;
use derive_deref::Deref;
use serde::Serialize;
use vecdb::{CheckedSub, PrintableIndex, StoredCompressed};
use zerocopy_derive::{FromBytes, Immutable, IntoBytes, KnownLayout};
use super::{
DateIndex, EmptyOutputIndex, Height, InputIndex, MonthIndex, OpReturnIndex, OutputIndex,
P2AAddressIndex, P2MSOutputIndex, P2PK33AddressIndex, P2PK65AddressIndex, P2PKHAddressIndex,
P2SHAddressIndex, P2TRAddressIndex, P2WPKHAddressIndex, P2WSHAddressIndex, TxIndex,
UnknownOutputIndex, YearIndex,
};
#[derive(
Debug,
Default,
Deref,
Clone,
Copy,
PartialEq,
Eq,
PartialOrd,
Ord,
FromBytes,
Immutable,
IntoBytes,
KnownLayout,
Serialize,
StoredCompressed,
Allocative,
)]
pub struct StoredU64(u64);
impl StoredU64 {
pub const ZERO: Self = Self(0);
pub fn new(counter: u64) -> Self {
Self(counter)
}
}
impl From<u64> for StoredU64 {
fn from(value: u64) -> Self {
Self(value)
}
}
impl From<StoredU64> for u64 {
fn from(value: StoredU64) -> Self {
value.0
}
}
impl From<StoredU64> for usize {
fn from(value: StoredU64) -> Self {
value.0 as usize
}
}
impl From<usize> for StoredU64 {
fn from(value: usize) -> Self {
Self(value as u64)
}
}
impl CheckedSub<StoredU64> for StoredU64 {
fn checked_sub(self, rhs: Self) -> Option<Self> {
self.0.checked_sub(rhs.0).map(Self)
}
}
impl Div<usize> for StoredU64 {
type Output = Self;
fn div(self, rhs: usize) -> Self::Output {
Self(self.0 / rhs as u64)
}
}
impl Add for StoredU64 {
type Output = Self;
fn add(self, rhs: Self) -> Self::Output {
Self(self.0 + rhs.0)
}
}
impl AddAssign for StoredU64 {
fn add_assign(&mut self, rhs: Self) {
*self = *self + rhs
}
}
impl From<f64> for StoredU64 {
fn from(value: f64) -> Self {
if value < 0.0 || value > u32::MAX as f64 {
panic!()
}
Self(value as u64)
}
}
impl From<StoredU64> for f64 {
fn from(value: StoredU64) -> Self {
value.0 as f64
}
}
impl From<TxIndex> for StoredU64 {
fn from(value: TxIndex) -> Self {
Self(*value as u64)
}
}
impl From<InputIndex> for StoredU64 {
fn from(value: InputIndex) -> Self {
Self(*value)
}
}
impl From<Height> for StoredU64 {
fn from(value: Height) -> Self {
Self(*value as u64)
}
}
impl From<OutputIndex> for StoredU64 {
fn from(value: OutputIndex) -> Self {
Self(*value)
}
}
impl From<DateIndex> for StoredU64 {
fn from(value: DateIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<MonthIndex> for StoredU64 {
fn from(value: MonthIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<YearIndex> for StoredU64 {
fn from(value: YearIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2PK65AddressIndex> for StoredU64 {
fn from(value: P2PK65AddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2PK33AddressIndex> for StoredU64 {
fn from(value: P2PK33AddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2PKHAddressIndex> for StoredU64 {
fn from(value: P2PKHAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<OpReturnIndex> for StoredU64 {
fn from(value: OpReturnIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2MSOutputIndex> for StoredU64 {
fn from(value: P2MSOutputIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2SHAddressIndex> for StoredU64 {
fn from(value: P2SHAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2WSHAddressIndex> for StoredU64 {
fn from(value: P2WSHAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2WPKHAddressIndex> for StoredU64 {
fn from(value: P2WPKHAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2TRAddressIndex> for StoredU64 {
fn from(value: P2TRAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<P2AAddressIndex> for StoredU64 {
fn from(value: P2AAddressIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<UnknownOutputIndex> for StoredU64 {
fn from(value: UnknownOutputIndex) -> Self {
Self::from(u64::from(value))
}
}
impl From<EmptyOutputIndex> for StoredU64 {
fn from(value: EmptyOutputIndex) -> Self {
Self::from(u64::from(value))
}
}
impl PrintableIndex for StoredU64 {
fn to_string() -> &'static str {
"u64"
}
fn to_possible_strings() -> &'static [&'static str] {
&["u64"]
}
}
impl std::fmt::Display for StoredU64 {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut buf = itoa::Buffer::new();
let str = buf.format(self.0);
f.write_str(str)
}
}