pub mod cost;
pub mod diff;
pub mod maintenance;
pub mod prelude;
pub mod workforce;
use crate::city::Stability;
use crate::infrastructure::mine::MineProduction;
use crate::infrastructure::storage::{OverallStorageCapacity, StorageCapacity};
use bon::Builder;
use derive_more::{Deref, Display, Into};
use diff::{FoodDiff, IronDiff, ResourcesDiff, StoneDiff, WoodDiff};
use nil_num::impl_mul_ceil;
use nil_num::ops::MulCeil;
use serde::{Deserialize, Serialize};
use std::cmp::Ordering;
use std::num::NonZeroU32;
use std::ops::{Add, AddAssign, Mul, MulAssign, Sub, SubAssign};
#[derive(Builder, Clone, Debug, PartialEq, Eq, Deserialize, Serialize)]
#[serde(default, rename_all = "camelCase")]
pub struct Resources {
#[builder(default)]
pub food: Food,
#[builder(default)]
pub iron: Iron,
#[builder(default)]
pub stone: Stone,
#[builder(default)]
pub wood: Wood,
}
impl Resources {
pub const MIN: Self = Self {
food: Food::MIN,
iron: Iron::MIN,
stone: Stone::MIN,
wood: Wood::MIN,
};
pub const MAX: Self = Self {
food: Food::MAX,
iron: Iron::MAX,
stone: Stone::MAX,
wood: Wood::MAX,
};
pub const PLAYER: Self = Self::splat(800);
pub const BOT: Self = Self::splat(2500);
pub const PRECURSOR: Self = Self::splat(5_000_000);
#[inline]
#[must_use]
pub fn new() -> Self {
Self::MIN.clone()
}
#[must_use]
pub const fn splat(value: u32) -> Self {
Self {
food: Food::new(value),
iron: Iron::new(value),
stone: Stone::new(value),
wood: Wood::new(value),
}
}
#[inline]
#[must_use]
pub fn with_food(&self, food: Food) -> Self {
Self { food, ..self.clone() }
}
#[inline]
#[must_use]
pub fn with_iron(&self, iron: Iron) -> Self {
Self { iron, ..self.clone() }
}
#[inline]
#[must_use]
pub fn with_stone(&self, stone: Stone) -> Self {
Self { stone, ..self.clone() }
}
#[inline]
#[must_use]
pub fn with_wood(&self, wood: Wood) -> Self {
Self { wood, ..self.clone() }
}
#[must_use]
pub fn silo(&self) -> Self {
Self {
food: self.food,
iron: Iron::MIN,
stone: Stone::MIN,
wood: Wood::MIN,
}
}
#[must_use]
pub fn warehouse(&self) -> Self {
Self {
food: Food::MIN,
iron: self.iron,
stone: self.stone,
wood: self.wood,
}
}
pub fn add_within_capacity(&mut self, diff: &ResourcesDiff, capacity: &OverallStorageCapacity) {
macro_rules! add {
($($resource:ident => $storage:ident),+ $(,)?) => {
$(
let resource = diff.$resource;
let storage = capacity.$storage;
self.$resource.add_within_capacity(resource, storage);
)+
};
}
add!(food => silo, iron => warehouse, stone => warehouse, wood => warehouse);
}
pub fn checked_sub(&self, rhs: &Self) -> Option<Self> {
Some(Self {
food: self.food.checked_sub(rhs.food)?,
iron: self.iron.checked_sub(rhs.iron)?,
stone: self.stone.checked_sub(rhs.stone)?,
wood: self.wood.checked_sub(rhs.wood)?,
})
}
pub fn sum(&self) -> u32 {
0u32
.saturating_add(self.food.0)
.saturating_add(self.iron.0)
.saturating_add(self.stone.0)
.saturating_add(self.wood.0)
}
#[inline]
pub fn sum_silo(&self) -> u32 {
self.silo().sum()
}
#[inline]
pub fn sum_warehouse(&self) -> u32 {
self.warehouse().sum()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.sum() == 0
}
}
impl Default for Resources {
fn default() -> Self {
Self::new()
}
}
impl Add for Resources {
type Output = Self;
fn add(self, rhs: Self) -> Self {
Self {
food: self.food + rhs.food,
iron: self.iron + rhs.iron,
stone: self.stone + rhs.stone,
wood: self.wood + rhs.wood,
}
}
}
impl Add<&Resources> for Resources {
type Output = Self;
fn add(self, rhs: &Resources) -> Self {
Self {
food: self.food + rhs.food,
iron: self.iron + rhs.iron,
stone: self.stone + rhs.stone,
wood: self.wood + rhs.wood,
}
}
}
impl AddAssign for Resources {
fn add_assign(&mut self, rhs: Self) {
*self = Self {
food: self.food + rhs.food,
iron: self.iron + rhs.iron,
stone: self.stone + rhs.stone,
wood: self.wood + rhs.wood,
};
}
}
impl AddAssign<&Resources> for Resources {
fn add_assign(&mut self, rhs: &Resources) {
*self = Self {
food: self.food + rhs.food,
iron: self.iron + rhs.iron,
stone: self.stone + rhs.stone,
wood: self.wood + rhs.wood,
};
}
}
impl Sub for Resources {
type Output = Self;
fn sub(self, rhs: Self) -> Self {
Self {
food: self.food - rhs.food,
iron: self.iron - rhs.iron,
stone: self.stone - rhs.stone,
wood: self.wood - rhs.wood,
}
}
}
impl Sub<&Resources> for Resources {
type Output = Self;
fn sub(self, rhs: &Resources) -> Self {
Self {
food: self.food - rhs.food,
iron: self.iron - rhs.iron,
stone: self.stone - rhs.stone,
wood: self.wood - rhs.wood,
}
}
}
impl SubAssign for Resources {
fn sub_assign(&mut self, rhs: Self) {
*self = Self {
food: self.food - rhs.food,
iron: self.iron - rhs.iron,
stone: self.stone - rhs.stone,
wood: self.wood - rhs.wood,
};
}
}
impl SubAssign<&Resources> for Resources {
fn sub_assign(&mut self, rhs: &Resources) {
*self = Self {
food: self.food - rhs.food,
iron: self.iron - rhs.iron,
stone: self.stone - rhs.stone,
wood: self.wood - rhs.wood,
};
}
}
impl Mul<u32> for &Resources {
type Output = Resources;
fn mul(self, rhs: u32) -> Self::Output {
Resources {
food: self.food * rhs,
iron: self.iron * rhs,
stone: self.stone * rhs,
wood: self.wood * rhs,
}
}
}
impl Mul<NonZeroU32> for &Resources {
type Output = Resources;
fn mul(self, rhs: NonZeroU32) -> Self::Output {
self * rhs.get()
}
}
macro_rules! decl_resource {
($($resource:ident),+ $(,)?) => {
paste::paste! {
$(
#[derive(
Clone,
Copy,
Debug,
Default,
Deref,
Display,
Into,
PartialEq,
Eq,
PartialOrd,
Ord,
Deserialize,
Serialize,
nil_num::F64Ops,
)]
#[into(u32, f64)]
pub struct $resource(u32);
impl $resource {
pub const MIN: Self = Self::new(0);
pub const MAX: Self = Self::new(u32::MAX);
#[inline]
pub const fn new(value: u32) -> Self {
Self(value)
}
#[inline]
pub fn checked_sub(self, rhs: Self) -> Option<Self> {
self.0.checked_sub(rhs.0).map(Self::new)
}
pub fn add_within_capacity(&mut self, diff: [<$resource Diff>], capacity: StorageCapacity) {
if diff < 0i32 {
*self += diff;
} else if self.0 < *capacity {
let capacity = $resource::from(capacity);
*self = (*self + diff).min(capacity);
}
}
}
impl From<u32> for $resource {
fn from(value: u32) -> Self {
Self(value)
}
}
impl From<f64> for $resource {
fn from(value: f64) -> Self {
debug_assert!(value.is_finite());
Self(value as u32)
}
}
impl From<MineProduction> for $resource {
fn from(value: MineProduction) -> Self {
Self(*value)
}
}
impl From<StorageCapacity> for $resource {
fn from(value: StorageCapacity) -> Self {
Self(*value)
}
}
impl PartialEq<u32> for $resource {
fn eq(&self, other: &u32) -> bool {
self.0.eq(other)
}
}
impl PartialOrd<u32> for $resource {
fn partial_cmp(&self, other: &u32) -> Option<Ordering> {
self.0.partial_cmp(other)
}
}
impl Add for $resource {
type Output = Self;
fn add(self, rhs: Self) -> Self {
Self(self.0.saturating_add(rhs.0))
}
}
impl Add<u32> for $resource {
type Output = Self;
fn add(self, rhs: u32) -> Self {
Self(self.0.saturating_add(rhs))
}
}
impl AddAssign for $resource {
fn add_assign(&mut self, rhs: Self) {
*self = *self + rhs;
}
}
impl Sub for $resource {
type Output = Self;
fn sub(self, rhs: Self) -> Self {
Self(self.0.saturating_sub(rhs.0))
}
}
impl Sub<u32> for $resource {
type Output = Self;
fn sub(self, rhs: u32) -> Self {
Self(self.0.saturating_sub(rhs))
}
}
impl SubAssign for $resource {
fn sub_assign(&mut self, rhs: Self) {
*self = *self - rhs;
}
}
impl Mul<u32> for $resource {
type Output = Self;
fn mul(self, rhs: u32) -> Self::Output {
Self(self.0.saturating_mul(rhs))
}
}
impl Mul<NonZeroU32> for $resource {
type Output = Self;
fn mul(self, rhs: NonZeroU32) -> Self::Output {
self * rhs.get()
}
}
impl Mul<Stability> for $resource {
type Output = $resource;
fn mul(self, rhs: Stability) -> Self::Output {
Self::from(self.mul_ceil(*rhs))
}
}
impl MulAssign<u32> for $resource {
fn mul_assign(&mut self, rhs: u32) {
*self = *self * rhs;
}
}
impl MulAssign<Stability> for $resource {
fn mul_assign(&mut self, rhs: Stability) {
*self = *self * rhs;
}
}
impl_mul_ceil!($resource);
)+
}
};
}
decl_resource!(Food, Iron, Stone, Wood);