| // This file is part of Eigen, a lightweight C++ template library | 
 | // for linear algebra. | 
 | // | 
 | // Copyright (C) 2008-2010 Gael Guennebaud <gael.guennebaud@inria.fr> | 
 | // | 
 | // This Source Code Form is subject to the terms of the Mozilla | 
 | // Public License v. 2.0. If a copy of the MPL was not distributed | 
 | // with this file, You can obtain one at http://mozilla.org/MPL/2.0/. | 
 |  | 
 | #ifndef EIGEN_SELECT_H | 
 | #define EIGEN_SELECT_H | 
 |  | 
 | // IWYU pragma: private | 
 | #include "./InternalHeaderCheck.h" | 
 |  | 
 | namespace Eigen { | 
 |  | 
 | /** \class Select | 
 |  * \ingroup Core_Module | 
 |  * | 
 |  * \brief Expression of a coefficient wise version of the C++ ternary operator ?: | 
 |  * | 
 |  * \tparam ConditionMatrixType the type of the \em condition expression which must be a boolean matrix | 
 |  * \tparam ThenMatrixType the type of the \em then expression | 
 |  * \tparam ElseMatrixType the type of the \em else expression | 
 |  * | 
 |  * This class represents an expression of a coefficient wise version of the C++ ternary operator ?:. | 
 |  * It is the return type of DenseBase::select() and most of the time this is the only way it is used. | 
 |  * | 
 |  * \sa DenseBase::select(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) const | 
 |  */ | 
 |  | 
 | namespace internal { | 
 | template <typename ConditionMatrixType, typename ThenMatrixType, typename ElseMatrixType> | 
 | struct traits<Select<ConditionMatrixType, ThenMatrixType, ElseMatrixType> > : traits<ThenMatrixType> { | 
 |   typedef typename traits<ThenMatrixType>::Scalar Scalar; | 
 |   typedef Dense StorageKind; | 
 |   typedef typename traits<ThenMatrixType>::XprKind XprKind; | 
 |   typedef typename ConditionMatrixType::Nested ConditionMatrixNested; | 
 |   typedef typename ThenMatrixType::Nested ThenMatrixNested; | 
 |   typedef typename ElseMatrixType::Nested ElseMatrixNested; | 
 |   enum { | 
 |     RowsAtCompileTime = ConditionMatrixType::RowsAtCompileTime, | 
 |     ColsAtCompileTime = ConditionMatrixType::ColsAtCompileTime, | 
 |     MaxRowsAtCompileTime = ConditionMatrixType::MaxRowsAtCompileTime, | 
 |     MaxColsAtCompileTime = ConditionMatrixType::MaxColsAtCompileTime, | 
 |     Flags = (unsigned int)ThenMatrixType::Flags & ElseMatrixType::Flags & RowMajorBit | 
 |   }; | 
 | }; | 
 | }  // namespace internal | 
 |  | 
 | template <typename ConditionMatrixType, typename ThenMatrixType, typename ElseMatrixType> | 
 | class Select : public internal::dense_xpr_base<Select<ConditionMatrixType, ThenMatrixType, ElseMatrixType> >::type, | 
 |                internal::no_assignment_operator { | 
 |  public: | 
 |   typedef typename internal::dense_xpr_base<Select>::type Base; | 
 |   EIGEN_DENSE_PUBLIC_INTERFACE(Select) | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC Select(const ConditionMatrixType& a_conditionMatrix, const ThenMatrixType& a_thenMatrix, | 
 |                                   const ElseMatrixType& a_elseMatrix) | 
 |       : m_condition(a_conditionMatrix), m_then(a_thenMatrix), m_else(a_elseMatrix) { | 
 |     eigen_assert(m_condition.rows() == m_then.rows() && m_condition.rows() == m_else.rows()); | 
 |     eigen_assert(m_condition.cols() == m_then.cols() && m_condition.cols() == m_else.cols()); | 
 |   } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC EIGEN_CONSTEXPR Index rows() const EIGEN_NOEXCEPT { return m_condition.rows(); } | 
 |   inline EIGEN_DEVICE_FUNC EIGEN_CONSTEXPR Index cols() const EIGEN_NOEXCEPT { return m_condition.cols(); } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC const Scalar coeff(Index i, Index j) const { | 
 |     if (m_condition.coeff(i, j)) | 
 |       return m_then.coeff(i, j); | 
 |     else | 
 |       return m_else.coeff(i, j); | 
 |   } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC const Scalar coeff(Index i) const { | 
 |     if (m_condition.coeff(i)) | 
 |       return m_then.coeff(i); | 
 |     else | 
 |       return m_else.coeff(i); | 
 |   } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC const ConditionMatrixType& conditionMatrix() const { return m_condition; } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC const ThenMatrixType& thenMatrix() const { return m_then; } | 
 |  | 
 |   inline EIGEN_DEVICE_FUNC const ElseMatrixType& elseMatrix() const { return m_else; } | 
 |  | 
 |  protected: | 
 |   typename ConditionMatrixType::Nested m_condition; | 
 |   typename ThenMatrixType::Nested m_then; | 
 |   typename ElseMatrixType::Nested m_else; | 
 | }; | 
 |  | 
 | /** \returns a matrix where each coefficient (i,j) is equal to \a thenMatrix(i,j) | 
 |  * if \c *this(i,j) != Scalar(0), and \a elseMatrix(i,j) otherwise. | 
 |  * | 
 |  * Example: \include MatrixBase_select.cpp | 
 |  * Output: \verbinclude MatrixBase_select.out | 
 |  * | 
 |  * \sa DenseBase::bitwiseSelect(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) | 
 |  */ | 
 | template <typename Derived> | 
 | template <typename ThenDerived, typename ElseDerived> | 
 | inline EIGEN_DEVICE_FUNC CwiseTernaryOp< | 
 |     internal::scalar_boolean_select_op<typename DenseBase<ThenDerived>::Scalar, typename DenseBase<ElseDerived>::Scalar, | 
 |                                        typename DenseBase<Derived>::Scalar>, | 
 |     ThenDerived, ElseDerived, Derived> | 
 | DenseBase<Derived>::select(const DenseBase<ThenDerived>& thenMatrix, const DenseBase<ElseDerived>& elseMatrix) const { | 
 |   using Op = internal::scalar_boolean_select_op<typename DenseBase<ThenDerived>::Scalar, | 
 |                                                 typename DenseBase<ElseDerived>::Scalar, Scalar>; | 
 |   return CwiseTernaryOp<Op, ThenDerived, ElseDerived, Derived>(thenMatrix.derived(), elseMatrix.derived(), derived(), | 
 |                                                                Op()); | 
 | } | 
 | /** Version of DenseBase::select(const DenseBase&, const DenseBase&) with | 
 |  * the \em else expression being a scalar value. | 
 |  * | 
 |  * \sa DenseBase::booleanSelect(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) const, class Select | 
 |  */ | 
 | template <typename Derived> | 
 | template <typename ThenDerived> | 
 | inline EIGEN_DEVICE_FUNC CwiseTernaryOp< | 
 |     internal::scalar_boolean_select_op<typename DenseBase<ThenDerived>::Scalar, typename DenseBase<ThenDerived>::Scalar, | 
 |                                        typename DenseBase<Derived>::Scalar>, | 
 |     ThenDerived, typename DenseBase<ThenDerived>::ConstantReturnType, Derived> | 
 | DenseBase<Derived>::select(const DenseBase<ThenDerived>& thenMatrix, | 
 |                            const typename DenseBase<ThenDerived>::Scalar& elseScalar) const { | 
 |   using ElseConstantType = typename DenseBase<ThenDerived>::ConstantReturnType; | 
 |   using Op = internal::scalar_boolean_select_op<typename DenseBase<ThenDerived>::Scalar, | 
 |                                                 typename DenseBase<ThenDerived>::Scalar, Scalar>; | 
 |   return CwiseTernaryOp<Op, ThenDerived, ElseConstantType, Derived>( | 
 |       thenMatrix.derived(), ElseConstantType(rows(), cols(), elseScalar), derived(), Op()); | 
 | } | 
 | /** Version of DenseBase::select(const DenseBase&, const DenseBase&) with | 
 |  * the \em then expression being a scalar value. | 
 |  * | 
 |  * \sa DenseBase::booleanSelect(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) const, class Select | 
 |  */ | 
 | template <typename Derived> | 
 | template <typename ElseDerived> | 
 | inline EIGEN_DEVICE_FUNC CwiseTernaryOp< | 
 |     internal::scalar_boolean_select_op<typename DenseBase<ElseDerived>::Scalar, typename DenseBase<ElseDerived>::Scalar, | 
 |                                        typename DenseBase<Derived>::Scalar>, | 
 |     typename DenseBase<ElseDerived>::ConstantReturnType, ElseDerived, Derived> | 
 | DenseBase<Derived>::select(const typename DenseBase<ElseDerived>::Scalar& thenScalar, | 
 |                            const DenseBase<ElseDerived>& elseMatrix) const { | 
 |   using ThenConstantType = typename DenseBase<ElseDerived>::ConstantReturnType; | 
 |   using Op = internal::scalar_boolean_select_op<typename DenseBase<ElseDerived>::Scalar, | 
 |                                                 typename DenseBase<ElseDerived>::Scalar, Scalar>; | 
 |   return CwiseTernaryOp<Op, ThenConstantType, ElseDerived, Derived>(ThenConstantType(rows(), cols(), thenScalar), | 
 |                                                                     elseMatrix.derived(), derived(), Op()); | 
 | } | 
 |  | 
 | }  // end namespace Eigen | 
 |  | 
 | #endif  // EIGEN_SELECT_H |