nil-core 0.5.4

Multiplayer strategy game
Documentation
// Copyright (C) Call of Nil contributors
// SPDX-License-Identifier: AGPL-3.0-only

use bon::Builder;
use rand::seq::IndexedRandom;
use serde::{Deserialize, Serialize};
use std::fmt::Debug;
use strum::{EnumIs, VariantArray};

#[derive(Builder, Clone, Debug, Deserialize, Serialize)]
#[derive_const(PartialEq, Eq)]
#[serde(rename_all = "camelCase")]
#[builder(const)]
#[cfg_attr(feature = "typescript", derive(ts_rs::TS))]
pub struct Ethics {
  power: EthicPowerAxis,
  truth: EthicTruthAxis,
}

impl Ethics {
  pub fn random() -> Self {
    Self {
      power: EthicPowerAxis::random(),
      truth: EthicTruthAxis::random(),
    }
  }

  #[inline]
  pub const fn power(&self) -> EthicPowerAxis {
    self.power
  }

  #[inline]
  pub const fn truth(&self) -> EthicTruthAxis {
    self.truth
  }

  #[inline]
  pub const fn is_militarist(&self) -> bool {
    self.power.is_militarist()
  }

  #[inline]
  pub const fn is_fanatic_militarist(&self) -> bool {
    self.power.is_fanatic_militarist()
  }

  #[inline]
  pub const fn is_pacifist(&self) -> bool {
    self.power.is_pacifist()
  }

  #[inline]
  pub const fn is_fanatic_pacifist(&self) -> bool {
    self.power.is_fanatic_pacifist()
  }

  #[inline]
  pub const fn is_materialist(&self) -> bool {
    self.truth.is_materialist()
  }

  #[inline]
  pub const fn is_fanatic_materialist(&self) -> bool {
    self.truth.is_fanatic_materialist()
  }

  #[inline]
  pub const fn is_spiritualist(&self) -> bool {
    self.truth.is_spiritualist()
  }

  #[inline]
  pub const fn is_fanatic_spiritualist(&self) -> bool {
    self.truth.is_fanatic_spiritualist()
  }
}

impl From<EthicPowerAxis> for Ethics {
  fn from(power: EthicPowerAxis) -> Self {
    Self {
      power,
      truth: EthicTruthAxis::random(),
    }
  }
}

impl From<EthicTruthAxis> for Ethics {
  fn from(truth: EthicTruthAxis) -> Self {
    Self {
      power: EthicPowerAxis::random(),
      truth,
    }
  }
}

#[derive(Clone, Copy, Debug, EnumIs, Deserialize, Serialize, VariantArray)]
#[derive_const(PartialEq, Eq)]
#[serde(rename_all = "kebab-case")]
#[cfg_attr(feature = "typescript", derive(ts_rs::TS))]
pub enum EthicPowerAxis {
  Militarist,
  FanaticMilitarist,
  Pacifist,
  FanaticPacifist,
}

impl EthicPowerAxis {
  pub fn random() -> Self {
    Self::VARIANTS
      .choose_weighted(&mut rand::rng(), |ethic| {
        match ethic {
          Self::Militarist | Self::Pacifist => 4u8,
          Self::FanaticMilitarist | Self::FanaticPacifist => 1u8,
        }
      })
      .copied()
      .expect("`Self::VARIANTS` should never be empty")
  }

  /// Whether this ethic is [`EthicPowerAxis::FanaticMilitarist`] or [`EthicPowerAxis::FanaticPacifist`].
  pub const fn is_fanatic(&self) -> bool {
    matches!(self, Self::FanaticMilitarist | Self::FanaticPacifist)
  }

  /// Whether this ethic is [`EthicPowerAxis::Militarist`] or [`EthicPowerAxis::FanaticMilitarist`].
  pub const fn is_militarist_variant(&self) -> bool {
    matches!(self, Self::Militarist | Self::FanaticMilitarist)
  }

  /// Whether this ethic is [`EthicPowerAxis::Pacifist`] or [`EthicPowerAxis::FanaticPacifist`].
  pub const fn is_pacifist_variant(&self) -> bool {
    matches!(self, Self::Pacifist | Self::FanaticPacifist)
  }

  pub const fn is_same_variant(&self, other: Self) -> bool {
    (self.is_militarist_variant() && other.is_militarist_variant())
      || (self.is_pacifist_variant() && other.is_pacifist_variant())
  }
}

#[derive(Clone, Copy, Debug, EnumIs, Deserialize, Serialize, VariantArray)]
#[derive_const(PartialEq, Eq)]
#[serde(rename_all = "kebab-case")]
#[cfg_attr(feature = "typescript", derive(ts_rs::TS))]
pub enum EthicTruthAxis {
  Materialist,
  FanaticMaterialist,
  Spiritualist,
  FanaticSpiritualist,
}

impl EthicTruthAxis {
  pub fn random() -> Self {
    Self::VARIANTS
      .choose_weighted(&mut rand::rng(), |ethic| {
        match ethic {
          Self::Materialist | Self::Spiritualist => 4u8,
          Self::FanaticMaterialist | Self::FanaticSpiritualist => 1u8,
        }
      })
      .copied()
      .expect("`Self::VARIANTS` should never be empty")
  }

  /// Whether this ethic is [`EthicTruthAxis::FanaticMaterialist`] or [`EthicTruthAxis::FanaticSpiritualist`].
  pub const fn is_fanatic(&self) -> bool {
    matches!(self, Self::FanaticMaterialist | Self::FanaticSpiritualist)
  }

  /// Whether this ethic is [`EthicTruthAxis::Materialist`] or [`EthicTruthAxis::FanaticMaterialist`].
  pub const fn is_materialist_variant(&self) -> bool {
    matches!(self, Self::Materialist | Self::FanaticMaterialist)
  }

  /// Whether this ethic is [`EthicTruthAxis::Spiritualist`] or [`EthicTruthAxis::FanaticSpiritualist`].
  pub const fn is_spiritualist_variant(&self) -> bool {
    matches!(self, Self::Spiritualist | Self::FanaticSpiritualist)
  }

  pub const fn is_same_variant(&self, other: Self) -> bool {
    (self.is_materialist_variant() && other.is_materialist_variant())
      || (self.is_spiritualist_variant() && other.is_spiritualist_variant())
  }
}