#include #include #include #include #include #include #include #include #include #include namespace Poincare { constexpr Expression::FunctionHelper MatrixIdentity::s_functionHelper; int MatrixIdentityNode::numberOfChildren() const { return MatrixIdentity::s_functionHelper.numberOfChildren(); } Expression MatrixIdentityNode::shallowReduce(ReductionContext reductionContext) { return MatrixIdentity(this).shallowReduce(reductionContext); } Layout MatrixIdentityNode::createLayout(Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const { return LayoutHelper::Prefix(MatrixIdentity(this), floatDisplayMode, numberOfSignificantDigits, MatrixIdentity::s_functionHelper.name()); } int MatrixIdentityNode::serialize(char * buffer, int bufferSize, Preferences::PrintFloatMode floatDisplayMode, int numberOfSignificantDigits) const { return SerializationHelper::Prefix(this, buffer, bufferSize, floatDisplayMode, numberOfSignificantDigits, MatrixIdentity::s_functionHelper.name()); } template Evaluation MatrixIdentityNode::templatedApproximate(ApproximationContext approximationContext) const { Evaluation input = childAtIndex(0)->approximate(T(), approximationContext); T r = input.toScalar(); // Undefined if the child is not real if (!std::isnan(r) && !std::isinf(r) && r > (T)0.0 // The child is defined and positive && std::ceil(r) == std::floor(r) // The child is an integer && r < ((float) INT_MAX)) // The child is not too big { return MatrixComplex::CreateIdentity((int)r); } return Complex::Undefined(); } Expression MatrixIdentity::shallowReduce(ExpressionNode::ReductionContext reductionContext) { { Expression e = Expression::defaultShallowReduce(); e = e.defaultHandleUnitsInChildren(); if (e.isUndefined()) { return e; } } Expression c = childAtIndex(0); if (c.type() != ExpressionNode::Type::Rational || !static_cast(c).isInteger()) { return *this; } Integer dimension = static_cast(c).signedIntegerNumerator(); if (dimension.isNegative() || dimension.isZero()) { return replaceWithUndefinedInPlace(); } if (Integer::NaturalOrder(dimension, Integer(15)) > 0) { // For now, after 15 it would fill the pool return *this; } int dim = dimension.extractedInt(); assert(dim != 0); Expression result = Matrix::CreateIdentity(dim); replaceWithInPlace(result); return result; } }