cómo averiguar correctamente el tipo de retorno de una lambda

Básicamente, ¿cómo hacer la siguiente compilación de código?

Sé que falló porque el compilador estaba tratando de evaluar algo como([](int &i){})(0) pero como resolver el problema?

template <class TElement>
struct foo {
    TElement _e;
    foo(TElement e) : _e(e){}
    template <class Lambda>
    void bar(Lambda f) {
        using TResult = decltype(std::declval<Lambda>()(std::declval<TElement>()));
    }
};

int main() {

    foo<int>(0).bar([](int i){}); // compile
    foo<int>(0).bar([](int &&i){}); // compile
    foo<int>(0).bar([](int const &i){}); // compile
    foo<int>(0).bar([](int &i){}); // failed

}