// Protocol Buffers - Google's data interchange format
// Copyright 2023 Google LLC.  All rights reserved.
//
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file or at
// https://developers.google.com/open-source/licenses/bsd

use std::fmt::Debug;

use crate::__internal::Private;
use crate::__runtime::InnerPrimitiveMut;
use crate::vtable::{PrimitiveWithRawVTable, ProxiedWithRawVTable, RawVTableOptionalMutatorData};
use crate::{Mut, MutProxy, Proxied, ProxiedWithPresence, SettableValue, View, ViewProxy};

/// A mutator for a primitive (numeric or enum) value of `T`.
///
/// This type is `protobuf::Mut<'msg, T>`.
pub struct PrimitiveMut<'msg, T> {
    inner: InnerPrimitiveMut<'msg, T>,
}

impl<'msg, T> Debug for PrimitiveMut<'msg, T>
where
    T: PrimitiveWithRawVTable,
{
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("PrimitiveMut").field("inner", &self.inner).finish()
    }
}

impl<'msg, T> PrimitiveMut<'msg, T> {
    /// # Safety
    /// `inner` must be valid and non-aliased for `T` for `'msg`
    #[doc(hidden)]
    pub unsafe fn from_inner(_private: Private, inner: InnerPrimitiveMut<'msg, T>) -> Self {
        Self { inner }
    }
}

// SAFETY: all `T` that can perform mutations don't mutate through a shared
// reference.
unsafe impl<'msg, T> Sync for PrimitiveMut<'msg, T> {}

impl<'msg, T> PrimitiveMut<'msg, T>
where
    T: PrimitiveWithRawVTable,
{
    /// Gets the current value of the field.
    pub fn get(&self) -> View<'_, T> {
        T::make_view(Private, self.inner)
    }

    /// Sets a new value for the field.
    pub fn set(&mut self, val: impl SettableValue<T>) {
        val.set_on(Private, self.as_mut())
    }

    #[doc(hidden)]
    pub fn set_primitive(&mut self, _private: Private, value: T) {
        // SAFETY: the raw mutator is valid for `'msg` as enforced by `Mut`
        unsafe { self.inner.set(value) }
    }
}

impl<'msg, T> ViewProxy<'msg> for PrimitiveMut<'msg, T>
where
    T: PrimitiveWithRawVTable,
{
    type Proxied = T;

    fn as_view(&self) -> View<'_, Self::Proxied> {
        self.get()
    }

    fn into_view<'shorter>(self) -> View<'shorter, Self::Proxied> {
        self.get()
    }
}

impl<'msg, T> MutProxy<'msg> for PrimitiveMut<'msg, T>
where
    T: PrimitiveWithRawVTable,
{
    fn as_mut(&mut self) -> Mut<'_, Self::Proxied> {
        PrimitiveMut { inner: self.inner }
    }

    fn into_mut<'shorter>(self) -> Mut<'shorter, Self::Proxied>
    where
        'msg: 'shorter,
    {
        self
    }
}

macro_rules! impl_singular_primitives {
  ($($t:ty),*) => {
      $(
        impl Proxied for $t {
            type View<'msg> = $t;
            type Mut<'msg> = PrimitiveMut<'msg, $t>;
        }

        impl<'msg> ViewProxy<'msg> for $t {
            type Proxied = $t;

            fn as_view(&self) -> View<'_, Self::Proxied> {
                *self
            }

            fn into_view<'shorter>(self) -> View<'shorter, Self::Proxied> {
                self
            }
        }

        impl SettableValue<$t> for $t {
            fn set_on<'msg>(self, private: Private, mut mutator: Mut<'msg, $t>) where $t: 'msg {
                mutator.set_primitive(private, self)
            }

            fn set_on_absent(
                self,
                _private: Private,
                absent_mutator: <$t as ProxiedWithPresence>::PresentMutData<'_>,
            ) -> <$t as ProxiedWithPresence>::AbsentMutData<'_>
            {
                absent_mutator.set(Private, self)
            }
        }

        impl ProxiedWithPresence for $t {
            type PresentMutData<'msg> = RawVTableOptionalMutatorData<'msg, $t>;
            type AbsentMutData<'msg> = RawVTableOptionalMutatorData<'msg, $t>;

            fn clear_present_field(
                present_mutator: Self::PresentMutData<'_>,
            ) -> Self::AbsentMutData<'_> {
                present_mutator.clear(Private)
            }

            fn set_absent_to_default(
                absent_mutator: Self::AbsentMutData<'_>,
            ) -> Self::PresentMutData<'_> {
                absent_mutator.set_absent_to_default(Private)
            }
        }

        impl PrimitiveWithRawVTable for $t {}

        // ProxiedInRepeated is implemented in {cpp,upb}.rs
      )*
  }
}

impl_singular_primitives!(bool, f32, f64, i32, i64, u32, u64);