como descobrir corretamente o tipo de retorno de um lambda

basicamente como fazer o seguinte código compilar?

Eu sei que falhou porque o compilador estava tentando avaliar algo como([](int &i){})(0) mas como resolver o 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

}

questionAnswers(3)

yourAnswerToTheQuestion