diff --git a/slides/properties.md b/slides/properties.md
index 81b522be4da2abf78bbb10aa3af7ba8a8d557356..fb8ab2a94e65f6863024d5e3656dfb74e3519faa 100644
--- a/slides/properties.md
+++ b/slides/properties.md
@@ -1,25 +1,42 @@
 ---
 title: The DuMuX property system
-subtitle: Flexible compile-time parameters
+subtitle: Flexible compile-time customization
 ---
 
-# Parameters vs. properties
+# Property System Design
+## Goals
 
-## Parameters vs. properties
+- Change parts of the simulation
+  - e.g. add a energy equation
+- Reuse specializations efficiently
+  - e.g. test various discretization schemes for same application
+- Group properties of the simulation
+  - Improves readability
+- Efficiency (runtime and implementation time)
 
-- <span style="color:#3498DB">Parameters</span> are set at __run-time__
-    <p>A default value may be used if the user does not provide one at run-time</p>
-- <span style="color:#3498DB">Properties</span> are known and set at __compile-time__
-    <p>
-        Can be used e.g. as template parameters (__types__ or values with `constexpr` specifier);
-        No run-time penalty, enable compiler to optimize
-    </p>
+## Customization points
 
+- Simulations have many possible customization points!
+  - fluid system, constitutive relations
+  - the type of the grid (incl. dimension)
+  - the terms in the equation
+  - discretization scheme
+  - details of the discretization scheme
+  - ...
+
+## Challenges
+
+- Simulations have many possible customization points
+  - This means many properties
+- Composing and reusing properties while keeping flexibility is technically challenging
+
+# A C++ solution
 ## Template parameters
 
 - C++ supports _generic programming_ via __templates__
-    - e.g. classes defined in terms of other types
-    - concrete versions of templates are stamped out upon compilation
+    - e.g. classes parameterized in terms of other types
+    - concrete versions of templates are defined by specifying concrete types
+    - the actual type has to be known at compilation time
 - __Flexible__: implementation not restricted to _concrete types_
 - __Efficient__: decisions made at compile-time
 
@@ -33,20 +50,12 @@ An example - `std::vector`
 template<typename T, typename A = std::allocator<T>>
 class vector;
 
-
 // Instantiation of a concrete vector - a vector of ints.
 // The compiler will define this concrete type for us,
 // using the definition of the class template.
 std::vector<int> v;
 ```
 
-## Template parameters
-
-An example - `std::vector`
-
-<img src="./img/template_example.png" width="800"/>
-
-
 ## Template specializations
 
 Template implementations can be specialized for concrete types
@@ -65,7 +74,23 @@ class MyVector<int>
 };
 ```
 
-## Too many template parameters
+## Templates in algorithms
+
+Using class templates, we can write _generic_ algorithms
+
+```cpp
+
+template<typename T>
+double two_norm(const MyVector<T>& v)
+{
+    double norm = 0.0;
+    for (int i = 0; i < v.size(); ++i)
+        norm += v[i]*v[i];
+    return std::sqrt(norm);
+}
+```
+
+## Challenge: Too many parameters
 
 For some classes, providing all template parameters can be very cumbersome and error-prone.
 
@@ -77,46 +102,30 @@ using GOF0 = Dune::GridOperator<
     CF, CF
 >;
 
-DGGO2 dggo2(gfs, cd, gfs, cf, lop, mbe);
 ```
 
-## Traits classes
+## Technique 1: Traits classes
 
 A usual way to group template parameters
 
 ```cpp
-struct MyGridOperatorTraits
-{
-    using FromGFS = ...;
-    // ...
-};
-
-using GOF0 = Dune::PDELab::GridOperator<MyGridOperatorTraits>;
-```
-
-## Inheriting from traits classes
-
-Inheritance may lead to unexpected results
-
-```cpp
-struct MyBaseTraits
-{
-    using Scalar = int;
-    using Vector = std::vector<Scalar>;
-};
-
-struct MyDoubleTraits : public MyBaseTraits
+template<class PV, class FSY, class FST, class SSY,...>
+struct TwoPVolumeVariablesTraits
 {
-    using Scalar = double;
+    using PrimaryVariables = PV;
+    using FluidSystem = FSY;
+    using FluidState = FST;
+    using SolidSystem = SSY;
+    using SolidState = SST;
+    using PermeabilityType = PT;
+    using ModelTraits = MT;
+    using SaturationReconstruction = SR;
 };
-
-// this is a vector of ints!
-typename MyDoubleTraits::Vector v{1.14142, 1.73205};
 ```
 
-## Type traits
+## Technique 2: Type traits
 
-Based on template specialization
+Why do we need the type?
 
 ```cpp
 // Type trait template declaration
@@ -131,34 +140,47 @@ template<typename T, int size>
 struct ValueType<Dune::FieldVector<T, size>> { using type = T; };
 ```
 
+## Technique 2: Type traits
+
+```cpp
+
+template <class Traits>
+class TwoPVolumeVariables
 
-## Type traits
+{
+   ...
+using FluidSystem = typename Traits::FluidSystem
+   ...
 
-A usage example
+```
 
+Usage: these VolumeVariables will work for various FluidSystems:
 
 ```cpp
-// expecting Container to export value_type
-template<typename Container>
-void someFunction(const Container& c) {
-    using V = typename Container::value_type;
-    // do something with V
-    // ...
-}
+Scalar mu = FluidSystem::viscosity(fluidState, paramCache, phaseIdx);
 ```
 
+## Challenge: Inheritance
+
+Inheritance may lead to unexpected results
+
 ```cpp
-// using type traits
-template<typename Container>
-void someFunction(const Container& c) {
-    using V = typename ValueType<Container>::type;
-    // do something with V
-    // ...
-}
+struct MyBaseTraits
+{
+    using Scalar = int;
+    using Vector = std::vector<Scalar>;
+};
+
+struct MyDoubleTraits : public MyBaseTraits
+{
+    using Scalar = double;
+};
+
+// this is a vector of ints!
+typename MyDoubleTraits::Vector v{1.14142, 1.73205};
 ```
 
 # The DuMuX Property System
-
 ## Property System Design
 
 - Based on __C++ template specialization__ (_type traits_)
@@ -194,7 +216,7 @@ A simplified example to illustrate the idea
 
 ```cpp
 template<class TypeTag>
-class GenericClass
+class GenericClass // e.g. LocalResidual
 {
     using PropA = typename Properties::PropTagA<TypeTag>::type;
     using PropB = typename Properties::PropTagB<TypeTag>::type;
@@ -236,8 +258,8 @@ struct Scalar<TypeTag, TTag::BaseTag> { using type = int; };
 // specialization of the Vector property for BaseTag
 template<class TypeTag>
 struct Vector<TypeTag, TTag::BaseTag> {
-    private: using Scalar = GetPropType<TypeTag, Properties::Scalar>;
-    public: using type = std::vector<Scalar>;
+    using Scalar = GetPropType<TypeTag, Properties::Scalar>;
+    using type = std::vector<Scalar>;
 };
 ```
 
@@ -413,3 +435,64 @@ class Problem
   * e.g. <span style="color:blue">TwoP, TwoPTwoC, TwoPNI</span>
 - By deriving your __type tag__ from those, your problem inherits all type information needed to set up the model at compile time!
 - Example: see Exercise
+
+# Alternatives to properties
+## Should I create new properties
+
+- Not necessary for most problems
+- Only when designing a complex class of models (e.g. porous medium flow models)
+- When there is many customization points
+
+## Alternative 1:
+
+- Use a simple template argument
+
+```cpp
+template <bool useNeummanBoundaryConditions>
+class Problem
+{
+    ...
+    if constexpr (useNeummanBoundaryConditions)
+        ...
+    else
+        ...
+};
+```
+
+## Alternative 2:
+
+C++ type traits
+
+```cpp
+template<class CouplingManager>
+struct SupportsMultithreadedAssembly
+: public std::false_type {}
+```
+
+## Alternative 2:
+
+C++ type traits
+
+```cpp
+class MyCouplingManager { ... };
+
+template<>
+struct SupportsMultithreadedAssembly<MyCouplingManager>
+: public std::true_type {}
+```
+
+## Alternative 2:
+
+Generic algorithm
+
+```cpp
+template<class CouplingManager>
+class Assembler {
+   void assemble() {
+      if (SupportsMultithreadedAssembly<CouplingManager>::value)
+        // assembly multi-threaded
+      else
+        // assembly single-threaded
+   }
+};
+```