diff --git a/dune/istl/umfpack.hh b/dune/istl/umfpack.hh
index 8fc9b8729e351d8b20d5b373b577cca6b74d8e28..59646276d1f69d93acb191ca1d722f70dde6ede6 100644
--- a/dune/istl/umfpack.hh
+++ b/dune/istl/umfpack.hh
@@ -172,18 +172,25 @@ namespace Dune {
   namespace Impl
   {
     template<class M, class = void>
-    struct UMFPackVectorChooser
-    {};
+    struct UMFPackVectorChooser;
+
+    /** @brief The type of the domain of the solver */
+    template<class M> using UMFPackDomainType = typename UMFPackVectorChooser<M>::domain_type;
+
+    /** @brief The type of the range of the solver */
+    template<class M> using UMFPackRangeType = typename UMFPackVectorChooser<M>::range_type;
 
     template<class M>
-    struct UMFPackVectorChooser<M, std::enable_if_t<(std::is_same<M,double>::value) || (std::is_same<M,std::complex<double> >::value)>>
+    struct UMFPackVectorChooser<M,
+      std::enable_if_t<(std::is_same<M,double>::value) || (std::is_same<M,std::complex<double> >::value)>>
     {
       using domain_type = M;
       using range_type  = M;
     };
 
     template<typename T, int n, int m>
-    struct UMFPackVectorChooser<FieldMatrix<T,n,m>>
+    struct UMFPackVectorChooser<FieldMatrix<T,n,m>,
+      std::enable_if_t<(std::is_same<T,double>::value) || (std::is_same<T,std::complex<double> >::value)>>
     {
       /** @brief The type of the domain of the solver */
       using domain_type = FieldVector<T,m>;
@@ -192,34 +199,35 @@ namespace Dune {
     };
 
     template<typename T, typename A>
-    struct UMFPackVectorChooser<BCRSMatrix<T,A> >
+    struct UMFPackVectorChooser<BCRSMatrix<T,A>,
+      std::void_t<UMFPackDomainType<T>, UMFPackRangeType<T>>>
     {
-      using sub_domain_type = typename UMFPackVectorChooser<T>::domain_type;
-      using sub_range_type = typename UMFPackVectorChooser<T>::range_type;
       /** @brief The type of the domain of the solver */
-      using domain_type = BlockVector<sub_domain_type, typename std::allocator_traits<A>::template rebind_alloc<sub_domain_type>>;
+      using domain_type = BlockVector<UMFPackDomainType<T>, typename std::allocator_traits<A>::template rebind_alloc<UMFPackDomainType<T>>>;
       /** @brief The type of the range of the solver */
-      using range_type  = BlockVector<sub_range_type, typename std::allocator_traits<A>::template rebind_alloc<sub_domain_type>>;
+      using range_type  = BlockVector<UMFPackRangeType<T>, typename std::allocator_traits<A>::template rebind_alloc<UMFPackRangeType<T>>>;
     };
 
     // to make the `UMFPackVectorChooser` work with `MultiTypeBlockMatrix`, we need to add an intermediate step for the rows, which are typically `MultiTypeBlockVector`
     template<typename FirstBlock, typename... Blocks>
-    struct UMFPackVectorChooser<MultiTypeBlockVector<FirstBlock, Blocks...> >
+    struct UMFPackVectorChooser<MultiTypeBlockVector<FirstBlock, Blocks...>,
+      std::void_t<UMFPackDomainType<FirstBlock>, UMFPackRangeType<FirstBlock>, UMFPackDomainType<Blocks>...>>
     {
       /** @brief The type of the domain of the solver */
-      using domain_type = MultiTypeBlockVector<typename UMFPackVectorChooser<FirstBlock>::domain_type,typename UMFPackVectorChooser<Blocks>::domain_type... >;
+      using domain_type = MultiTypeBlockVector<UMFPackDomainType<FirstBlock>, UMFPackDomainType<Blocks>...>;
       /** @brief The type of the range of the solver */
-      using range_type  = typename UMFPackVectorChooser<FirstBlock>::range_type;
+      using range_type  = UMFPackRangeType<FirstBlock>;
     };
 
     // specialization for `MultiTypeBlockMatrix` with `MultiTypeBlockVector` rows
     template<typename FirstRow, typename... Rows>
-    struct UMFPackVectorChooser<MultiTypeBlockMatrix<FirstRow, Rows...> >
+    struct UMFPackVectorChooser<MultiTypeBlockMatrix<FirstRow, Rows...>,
+      std::void_t<UMFPackDomainType<FirstRow>, UMFPackRangeType<FirstRow>, UMFPackRangeType<Rows>...>>
     {
       /** @brief The type of the domain of the solver */
-      using domain_type = typename UMFPackVectorChooser<FirstRow>::domain_type;
+      using domain_type = UMFPackDomainType<FirstRow>;
       /** @brief The type of the range of the solver */
-      using range_type  = MultiTypeBlockVector< typename UMFPackVectorChooser<FirstRow>::range_type, typename UMFPackVectorChooser<Rows>::range_type... >;
+      using range_type  = MultiTypeBlockVector< UMFPackRangeType<FirstRow>, UMFPackRangeType<Rows>... >;
     };
 
     // dummy class to represent no BitVector
@@ -242,15 +250,11 @@ namespace Dune {
    *
    * \note This will only work if dune-istl has been configured to use UMFPack
    */
-  template<typename M>
-  class UMFPack
-      : public InverseOperator<
-          typename Impl::UMFPackVectorChooser<M>::domain_type,
-          typename Impl::UMFPackVectorChooser<M>::range_type >
+  template<typename M, typename D = Impl::UMFPackDomainType<M>, typename R = Impl::UMFPackRangeType<M>>
+  class UMFPack : public InverseOperator<D,R>
   {
     using T = typename M::field_type;
 
-
     public:
     using size_type = SuiteSparse_long;
 
@@ -262,9 +266,9 @@ namespace Dune {
     /** @brief Type of an associated initializer class. */
     using MatrixInitializer = ISTL::Impl::BCCSMatrixInitializer<M, size_type>;
     /** @brief The type of the domain of the solver. */
-    using domain_type = typename Impl::UMFPackVectorChooser<M>::domain_type;
+    using domain_type = D;
     /** @brief The type of the range of the solver. */
-    using range_type = typename Impl::UMFPackVectorChooser<M>::range_type;
+    using range_type = R;
 
     //! Category of the solver (see SolverCategory::Category)
     virtual SolverCategory::Category category() const
@@ -789,18 +793,15 @@ namespace Dune {
 
   struct UMFPackCreator {
 
-    template<class M> using DomainType = typename Impl::UMFPackVectorChooser<M>::domain_type;
-    template<class M> using RangeType = typename Impl::UMFPackVectorChooser<M>::range_type;
-
     template<class TL, class M,class=void> struct isValidBlock : std::false_type{};
     template<class TL, class M> struct isValidBlock<TL,M,
       std::enable_if_t<
-           std::is_same_v<DomainType<M>, typename Dune::TypeListElement<1,TL>::type>
-        && std::is_same_v<RangeType<M>,  typename Dune::TypeListElement<2,TL>::type>
+           std::is_same_v<Impl::UMFPackDomainType<M>, typename Dune::TypeListElement<1,TL>::type>
+        && std::is_same_v<Impl::UMFPackRangeType<M>,  typename Dune::TypeListElement<2,TL>::type>
       >> : std::true_type {};
 
     template<typename TL, typename M>
-    std::shared_ptr<Dune::InverseOperator<DomainType<M>,RangeType<M>>>
+    std::shared_ptr<Dune::InverseOperator<Impl::UMFPackDomainType<M>,Impl::UMFPackRangeType<M>>>
     operator() (TL /*tl*/, const M& mat, const Dune::ParameterTree& config,
       std::enable_if_t<isValidBlock<TL, M>::value,int> = 0) const
     {
@@ -817,8 +818,8 @@ namespace Dune {
     {
       using D = typename Dune::TypeListElement<1,TL>::type;
       using R = typename Dune::TypeListElement<2,TL>::type;
-      using DU = Std::detected_t<DomainType, M>;
-      using RU = Std::detected_t<RangeType, M>;
+      using DU = Std::detected_t< Impl::UMFPackDomainType, M>;
+      using RU = Std::detected_t< Impl::UMFPackRangeType, M>;
       DUNE_THROW(UnsupportedType,
         "Unsupported Types in UMFPack:\n"
         "Matrix: " << className<M>() << ""