fyrox-ui 1.0.1

Extendable UI library
Documentation
// Copyright (c) 2019-present Dmitry Stepanov and Fyrox Engine contributors.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.

use crate::{
    core::{algebra::SMatrix, num_traits::NumCast},
    inspector::{
        editors::{
            PropertyEditorBuildContext, PropertyEditorDefinition, PropertyEditorInstance,
            PropertyEditorMessageContext, PropertyEditorTranslationContext,
        },
        FieldAction, InspectorError, PropertyChanged,
    },
    matrix::{MatrixEditorBuilder, MatrixEditorMessage},
    message::{MessageDirection, UiMessage},
    numeric::NumericType,
    widget::WidgetBuilder,
    Thickness,
};
use std::{any::TypeId, marker::PhantomData};

#[derive(Debug)]
pub struct MatrixPropertyEditorDefinition<const R: usize, const C: usize, T: NumericType> {
    pub phantom: PhantomData<T>,
}

impl<const R: usize, const C: usize, T: NumericType> Default
    for MatrixPropertyEditorDefinition<R, C, T>
{
    fn default() -> Self {
        Self {
            phantom: PhantomData,
        }
    }
}

impl<const R: usize, const C: usize, T: NumericType> PropertyEditorDefinition
    for MatrixPropertyEditorDefinition<R, C, T>
{
    fn value_type_id(&self) -> TypeId {
        TypeId::of::<SMatrix<T, R, C>>()
    }

    fn create_instance(
        &self,
        ctx: PropertyEditorBuildContext,
    ) -> Result<PropertyEditorInstance, InspectorError> {
        let value = ctx.property_info.cast_value::<SMatrix<T, R, C>>()?;
        Ok(PropertyEditorInstance::simple(
            MatrixEditorBuilder::new(WidgetBuilder::new().with_margin(Thickness::uniform(1.0)))
                .with_min(SMatrix::repeat(
                    ctx.property_info
                        .min_value
                        .and_then(NumCast::from)
                        .unwrap_or_else(T::min_value),
                ))
                .with_max(SMatrix::repeat(
                    ctx.property_info
                        .max_value
                        .and_then(NumCast::from)
                        .unwrap_or_else(T::max_value),
                ))
                .with_step(SMatrix::repeat(
                    ctx.property_info
                        .step
                        .and_then(NumCast::from)
                        .unwrap_or_else(T::one),
                ))
                .with_value(*value)
                .build(ctx.build_context),
        ))
    }

    fn create_message(
        &self,
        ctx: PropertyEditorMessageContext,
    ) -> Result<Option<UiMessage>, InspectorError> {
        let value = ctx.property_info.cast_value::<SMatrix<T, R, C>>()?;
        Ok(Some(UiMessage::for_widget(
            ctx.instance,
            MatrixEditorMessage::Value(*value),
        )))
    }

    fn translate_message(&self, ctx: PropertyEditorTranslationContext) -> Option<PropertyChanged> {
        if ctx.message.direction() == MessageDirection::FromWidget {
            if let Some(MatrixEditorMessage::Value(value)) =
                ctx.message.data::<MatrixEditorMessage<R, C, T>>()
            {
                return Some(PropertyChanged {
                    name: ctx.name.to_string(),
                    action: FieldAction::object(*value),
                });
            }
        }
        None
    }
}