| // 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 | 
 |  | 
 | #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 | 
 |   }; | 
 | }; | 
 | } | 
 |  | 
 | 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), and \a elseMatrix(i,j) otherwise. | 
 |   * | 
 |   * Example: \include MatrixBase_select.cpp | 
 |   * Output: \verbinclude MatrixBase_select.out | 
 |   * | 
 |   * \sa class Select | 
 |   */ | 
 | template<typename Derived> | 
 | template<typename ThenDerived,typename ElseDerived> | 
 | inline EIGEN_DEVICE_FUNC const Select<Derived,ThenDerived,ElseDerived> | 
 | DenseBase<Derived>::select(const DenseBase<ThenDerived>& thenMatrix, | 
 |                             const DenseBase<ElseDerived>& elseMatrix) const | 
 | { | 
 |   return Select<Derived,ThenDerived,ElseDerived>(derived(), thenMatrix.derived(), elseMatrix.derived()); | 
 | } | 
 |  | 
 | /** Version of DenseBase::select(const DenseBase&, const DenseBase&) with | 
 |   * the \em else expression being a scalar value. | 
 |   * | 
 |   * \sa DenseBase::select(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) const, class Select | 
 |   */ | 
 | template<typename Derived> | 
 | template<typename ThenDerived> | 
 | inline EIGEN_DEVICE_FUNC const Select<Derived,ThenDerived, typename ThenDerived::ConstantReturnType> | 
 | DenseBase<Derived>::select(const DenseBase<ThenDerived>& thenMatrix, | 
 |                            const typename ThenDerived::Scalar& elseScalar) const | 
 | { | 
 |   return Select<Derived,ThenDerived,typename ThenDerived::ConstantReturnType>( | 
 |     derived(), thenMatrix.derived(), ThenDerived::Constant(rows(),cols(),elseScalar)); | 
 | } | 
 |  | 
 | /** Version of DenseBase::select(const DenseBase&, const DenseBase&) with | 
 |   * the \em then expression being a scalar value. | 
 |   * | 
 |   * \sa DenseBase::select(const DenseBase<ThenDerived>&, const DenseBase<ElseDerived>&) const, class Select | 
 |   */ | 
 | template<typename Derived> | 
 | template<typename ElseDerived> | 
 | inline EIGEN_DEVICE_FUNC const Select<Derived, typename ElseDerived::ConstantReturnType, ElseDerived > | 
 | DenseBase<Derived>::select(const typename ElseDerived::Scalar& thenScalar, | 
 |                            const DenseBase<ElseDerived>& elseMatrix) const | 
 | { | 
 |   return Select<Derived,typename ElseDerived::ConstantReturnType,ElseDerived>( | 
 |     derived(), ElseDerived::Constant(rows(),cols(),thenScalar), elseMatrix.derived()); | 
 | } | 
 |  | 
 | } // end namespace Eigen | 
 |  | 
 | #endif // EIGEN_SELECT_H |