diff --git a/dumux/porousmediumflow/richards/model.hh b/dumux/porousmediumflow/richards/model.hh
index 388546a28234cc1e618d4e04801de23151956501..8a080f62d984dcd8003475c6c1692841aa89b49f 100644
--- a/dumux/porousmediumflow/richards/model.hh
+++ b/dumux/porousmediumflow/richards/model.hh
@@ -108,7 +108,6 @@
 #include "indices.hh"
 #include "volumevariables.hh"
 #include "vtkoutputfields.hh"
-#include "newtoncontroller.hh"
 #include "localresidual.hh"
 #include "primaryvariableswitch.hh"
 
diff --git a/dumux/porousmediumflow/richards/newtonsolver.hh b/dumux/porousmediumflow/richards/newtonsolver.hh
new file mode 100644
index 0000000000000000000000000000000000000000..64541cbb08e7f43d15d54ea4a832d202065f4b06
--- /dev/null
+++ b/dumux/porousmediumflow/richards/newtonsolver.hh
@@ -0,0 +1,127 @@
+// -*- mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
+// vi: set et ts=4 sw=4 sts=4:
+/*****************************************************************************
+ *   See the file COPYING for full copying permissions.                      *
+ *                                                                           *
+ *   This program is free software: you can redistribute it and/or modify    *
+ *   it under the terms of the GNU General Public License as published by    *
+ *   the Free Software Foundation, either version 2 of the License, or       *
+ *   (at your option) any later version.                                     *
+ *                                                                           *
+ *   This program is distributed in the hope that it will be useful,         *
+ *   but WITHOUT ANY WARRANTY; without even the implied warranty of          *
+ *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the            *
+ *   GNU General Public License for more details.                            *
+ *                                                                           *
+ *   You should have received a copy of the GNU General Public License       *
+ *   along with this program.  If not, see <http://www.gnu.org/licenses/>.   *
+ *****************************************************************************/
+/*!
+ * \file
+ * \ingroup RichardsModel
+ * \brief A Richards model newton solver.
+ */
+#ifndef DUMUX_RICHARDS_NEWTON_SOLVER_HH
+#define DUMUX_RICHARDS_NEWTON_SOLVER_HH
+
+#include <dumux/common/properties.hh>
+#include <dumux/nonlinear/newtonsolver.hh>
+
+namespace Dumux {
+/*!
+ * \ingroup RichardsModel
+ * \brief A Richards model specific newton solver.
+ *
+ * This controller 'knows' what a 'physically meaningful' solution is
+ * and can thus do update smarter than the plain Newton solver.
+ *
+ * \todo make this typetag independent by extracting anything model specific from assembler
+ *       or from possible ModelTraits.
+ */
+template <class TypeTag, class Assembler, class LinearSolver>
+class RichardsNewtonSolver : public NewtonSolver<Assembler, LinearSolver>
+{
+    using Scalar = typename Assembler::Scalar;
+    using ParentType = NewtonSolver<Assembler, LinearSolver>;
+    using SolutionVector = typename Assembler::ResidualType;
+
+    using MaterialLaw = typename GET_PROP_TYPE(TypeTag, MaterialLaw);
+    using ElementSolution =  typename GET_PROP_TYPE(TypeTag, ElementSolutionVector);
+    using Indices = typename GET_PROP_TYPE(TypeTag, Indices);
+    enum { pressureIdx = Indices::pressureIdx };
+
+public:
+    using ParentType::ParentType;
+
+private:
+
+    /*!
+     * \brief Update the current solution of the newton method
+     *
+     * \todo TODO: doc me!
+     *
+     * \param uCurrentIter The solution after the current Newton iteration \f$ u^{k+1} \f$
+     * \param uLastIter The solution after the last Newton iteration \f$ u^k \f$
+     * \param deltaU The vector of differences between the last
+     *               iterative solution and the next one \f$ \Delta u^k \f$
+     */
+    void choppedUpdate_(SolutionVector &uCurrentIter,
+                        const SolutionVector &uLastIter,
+                        const SolutionVector &deltaU) final
+    {
+        uCurrentIter = uLastIter;
+        uCurrentIter -= deltaU;
+
+        // do not clamp anything after 5 iterations
+        if (this->numSteps_ <= 4)
+        {
+            // clamp saturation change to at most 20% per iteration
+            const auto& fvGridGeometry = this->assembler().fvGridGeometry();
+            for (const auto& element : elements(fvGridGeometry.gridView()))
+            {
+                auto fvGeometry = localView(fvGridGeometry);
+                fvGeometry.bindElement(element);
+
+                for (auto&& scv : scvs(fvGeometry))
+                {
+                    auto dofIdxGlobal = scv.dofIndex();
+
+                    // calculate the old wetting phase saturation
+                    const auto& spatialParams = this->assembler().problem().spatialParams();
+                    const ElementSolution elemSol(element, uCurrentIter, fvGridGeometry);
+                    const auto& materialLawParams = spatialParams.materialLawParams(element, scv, elemSol);
+                    const Scalar pcMin = MaterialLaw::pc(materialLawParams, 1.0);
+                    const Scalar pw = uLastIter[dofIdxGlobal][pressureIdx];
+                    using std::max;
+                    const Scalar pn = max(this->assembler().problem().nonWettingReferencePressure(), pw + pcMin);
+                    const Scalar pcOld = pn - pw;
+                    const Scalar SwOld = max(0.0, MaterialLaw::sw(materialLawParams, pcOld));
+
+                    // convert into minimum and maximum wetting phase
+                    // pressures
+                    const Scalar pwMin = pn - MaterialLaw::pc(materialLawParams, SwOld - 0.2);
+                    const Scalar pwMax = pn - MaterialLaw::pc(materialLawParams, SwOld + 0.2);
+
+                    // clamp the result
+                    using std::min; using std::max;
+                    uCurrentIter[dofIdxGlobal][pressureIdx] = max(pwMin, min(uCurrentIter[dofIdxGlobal][pressureIdx], pwMax));
+                }
+            }
+        }
+
+        if (this->enableResidualCriterion())
+            this->computeResidualReduction_(uCurrentIter);
+
+        else
+        {
+            // If we get here, the convergence criterion does not require
+            // additional residual evalutions. Thus, the grid variables have
+            // not yet been updated to the new uCurrentIter.
+            this->assembler().updateGridVariables(uCurrentIter);
+        }
+    }
+};
+
+} // end namespace Dumux
+
+#endif
diff --git a/test/porousmediumflow/richards/implicit/test_richardslens_fv.cc b/test/porousmediumflow/richards/implicit/test_richardslens_fv.cc
index f43e186ec5e29bcf78a390a27ac9b3afaa67ef9e..c108a3c94e90da4bcdae6cdfbb6b736fa34e0519 100644
--- a/test/porousmediumflow/richards/implicit/test_richardslens_fv.cc
+++ b/test/porousmediumflow/richards/implicit/test_richardslens_fv.cc
@@ -41,9 +41,7 @@
 #include <dumux/common/defaultusagemessage.hh>
 
 #include <dumux/linear/amgbackend.hh>
-#include <dumux/nonlinear/newtonmethod.hh>
-#include <dumux/nonlinear/newtoncontroller.hh>
-#include <dumux/porousmediumflow/richards/newtoncontroller.hh>
+#include <dumux/porousmediumflow/richards/newtonsolver.hh>
 
 #include <dumux/assembly/fvassembler.hh>
 
@@ -157,10 +155,8 @@ int main(int argc, char** argv) try
     auto linearSolver = std::make_shared<LinearSolver>(leafGridView, fvGridGeometry->dofMapper());
 
     // the non-linear solver
-    using NewtonController = Dumux::RichardsNewtonController<TypeTag>;
-    using NewtonMethod = Dumux::NewtonMethod<NewtonController, Assembler, LinearSolver>;
-    auto newtonController = std::make_shared<NewtonController>(timeLoop);
-    NewtonMethod nonLinearSolver(newtonController, assembler, linearSolver);
+    using NewtonSolver = Dumux::RichardsNewtonSolver<TypeTag, Assembler, LinearSolver>;
+    NewtonSolver nonLinearSolver(assembler, linearSolver, timeLoop);
 
     // time loop
     timeLoop->start(); do
@@ -201,7 +197,7 @@ int main(int argc, char** argv) try
         timeLoop->reportTimeStep();
 
         // set new dt as suggested by newton controller
-        timeLoop->setTimeStepSize(newtonController->suggestTimeStepSize(timeLoop->timeStepSize()));
+        timeLoop->setTimeStepSize(nonLinearSolver.suggestTimeStepSize(timeLoop->timeStepSize()));
 
     } while (!timeLoop->finished());