#ifndef COMMON_ALGORITHM_FOREACH_H
#define COMMON_ALGORITHM_FOREACH_H
#include <memory>
/*
Calls the function 'callback' on every element in the container.
Example usage:
std::vector<Element> array;
ForEach(array, DoSomething);
Or, pass arguments in:
ForEach(array, DoSomething, 17, "Text");
*/
//Takes a container and passes each element to a lambda or function.
//Version for functors or non-generic lambdas (i.e. lambdas with capture lists).
template<typename ContainerType, typename CallbackFunc, typename ...Args>
void ForEach(ContainerType &container, const CallbackFunc &callback, Args&& ...args)
{
for(auto &element : container)
{
callback(element, args...);
}
}
//Const version, for functors and lambdas.
template<typename ContainerType, typename CallbackFunc, typename ...Args>
void ForEach(const ContainerType &container, CallbackFunc callback, Args&& ...args)
{
for(auto &element : container)
{
callback(element, args...);
}
}
//============================================================================
/*
Calls the member-function 'member_func' on every element in the container.
Example usage:
std::vector<Element> array;
ForEach(array, &Element::DoSomething);
Or, pass arguments in:
ForEach(array, &Element::DoSomething, 17, "Text");
*/
//Calls the member-function 'member_func' on every element in the container, optionally passing in arguments.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEach(ContainerType &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
for(auto &element : container)
{
element.*member_func(args...);
}
}
//Const version of 'ForEach()' where you can pass arguments into the member-function call.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEach(const ContainerType &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
for(const auto &element : container)
{
(element.*member_func)(args...);
}
}
//Calls the member-function 'member_func' on every element in the container, optionally passing in arguments.
//Also, returns true if any of the member-functions return true.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
bool ForEach_Return(ContainerType &container, bool (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
bool result = false;
for(auto &element : container)
{
result = (result || (element.*member_func)(args...));
}
return result;
}
//Const version.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
bool ForEach_Return(const ContainerType &container, bool (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
bool result = false;
for(const auto &element : container)
{
result = (result || (element.*member_func)(args...));
}
return result;
}
//============================================================================
//Wrappers for std::unique_ptr and std::shared_ptr, with callback-function calls.
template<typename Container, typename CallbackFunc, typename ...Args>
void priv_ForEachPtr_impl_forSharedOrUniquePtrsCallbackFunc(Container &container, CallbackFunc callback, Args&& ...args)
{
for(auto &ptr : container)
{
if(ptr)
{
callback(*ptr, args...);
}
}
}
//Wrappers for std::weak_ptr, with callback-function calls.
template<typename Container, typename CallbackFunc, typename ...Args>
void priv_ForEachPtr_impl_forWeakPtrsCallbackFunc(Container &container, CallbackFunc callback, Args&& ...args)
{
for(auto &weak_ptr : container)
{
auto shared_ptr = weak_ptr.lock();
if(shared_ptr)
{
callback(*shared_ptr, args...);
}
}
}
//============================================================================
//Wrappers for std::unique_ptr and std::shared_ptr, with member-function calls.
template<typename Container, typename MemberFunc, typename ...Args>
void priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(Container &container, MemberFunc member_func, Args&& ...args)
{
for(auto &ptr : container)
{
if(ptr)
{
((*ptr).*member_func)(args...);
}
}
}
//Wrappers for std::weak_ptr, with member-function calls.
template<typename Container, typename MemberFunc, typename ...Args>
void priv_ForEachPtr_impl_forWeakPtrsMemberFunc(Container &container, MemberFunc member_func, Args&& ...args)
{
for(auto &weak_ptr : container)
{
auto shared_ptr = weak_ptr.lock();
if(shared_ptr)
{
((*shared_ptr).*member_func)(args...);
}
}
}
//============================================================================
//Overload for vectors of unique_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
//============================================================================
//Overload for vectors of shared_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
//============================================================================
//Overload for vectors of weak_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
//============================================================================
#endif // COMMON_ALGORITHM_FOREACH_H
#ifndef COMMON_ALGORITHM_FOREACH_H
#define COMMON_ALGORITHM_FOREACH_H

#include <memory>

/*
	Calls the function 'callback' on every element in the container.

	Example usage:

		std::vector<Element> array;
		ForEach(array, DoSomething);

	Or, pass arguments in:

		ForEach(array, DoSomething, 17, "Text");
*/

//Takes a container and passes each element to a lambda or function.
//Version for functors or non-generic lambdas (i.e. lambdas with capture lists).
template<typename ContainerType, typename CallbackFunc, typename ...Args>
void ForEach(ContainerType &container, const CallbackFunc &callback, Args&& ...args)
{
	for(auto &element : container)
	{
		callback(element, args...);
	}
}

//Const version, for functors and lambdas.
template<typename ContainerType, typename CallbackFunc, typename ...Args>
void ForEach(const ContainerType &container, CallbackFunc callback, Args&& ...args)
{
	for(auto &element : container)
	{
		callback(element, args...);
	}
}

//============================================================================

/*
	Calls the member-function 'member_func' on every element in the container.

	Example usage:

		std::vector<Element> array;
		ForEach(array, &Element::DoSomething);

	Or, pass arguments in:

		ForEach(array, &Element::DoSomething, 17, "Text");
*/

//Calls the member-function 'member_func' on every element in the container, optionally passing in arguments.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEach(ContainerType &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
	for(auto &element : container)
	{
		element.*member_func(args...);
	}
}

//Const version of 'ForEach()' where you can pass arguments into the member-function call.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEach(const ContainerType &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
	for(const auto &element : container)
	{
		(element.*member_func)(args...);
	}
}

//Calls the member-function 'member_func' on every element in the container, optionally passing in arguments.
//Also, returns true if any of the member-functions return true.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
bool ForEach_Return(ContainerType &container, bool (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
	bool result = false;
	for(auto &element : container)
	{
		result = (result || (element.*member_func)(args...));
	}

	return result;
}

//Const version.
template<typename ContainerType, typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
bool ForEach_Return(const ContainerType &container, bool (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
	bool result = false;
	for(const auto &element : container)
	{
		result = (result || (element.*member_func)(args...));
	}

	return result;
}

//============================================================================

//Wrappers for std::unique_ptr and std::shared_ptr, with callback-function calls.
template<typename Container, typename CallbackFunc, typename ...Args>
void priv_ForEachPtr_impl_forSharedOrUniquePtrsCallbackFunc(Container &container, CallbackFunc callback, Args&& ...args)
{
	for(auto &ptr : container)
	{
		if(ptr)
		{
			callback(*ptr, args...);
		}
	}
}

//Wrappers for std::weak_ptr, with callback-function calls.
template<typename Container, typename CallbackFunc, typename ...Args>
void priv_ForEachPtr_impl_forWeakPtrsCallbackFunc(Container &container, CallbackFunc callback, Args&& ...args)
{
	for(auto &weak_ptr : container)
	{
		auto shared_ptr = weak_ptr.lock();

		if(shared_ptr)
		{
			callback(*shared_ptr, args...);
		}
	}
}

//============================================================================

//Wrappers for std::unique_ptr and std::shared_ptr, with member-function calls.
template<typename Container, typename MemberFunc, typename ...Args>
void priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(Container &container, MemberFunc member_func, Args&& ...args)
{
	for(auto &ptr : container)
	{
		if(ptr)
		{
			((*ptr).*member_func)(args...);
		}
	}
}

//Wrappers for std::weak_ptr, with member-function calls.
template<typename Container, typename MemberFunc, typename ...Args>
void priv_ForEachPtr_impl_forWeakPtrsMemberFunc(Container &container, MemberFunc member_func, Args&& ...args)
{
	for(auto &weak_ptr : container)
	{
		auto shared_ptr = weak_ptr.lock();

		if(shared_ptr)
		{
			((*shared_ptr).*member_func)(args...);
		}
	}
}

//============================================================================

//Overload for vectors of unique_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}

//============================================================================

//Overload for vectors of shared_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}

//============================================================================

//Overload for vectors of weak_ptr.
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}
template<typename ElementType, typename ...MembFuncArgs, typename ...ParamArgs>
void ForEachPtr(const std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
{
	priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
}

//============================================================================

#endif // COMMON_ALGORITHM_FOREACH_H
prog.cpp:166:22: error: variable or field 'ForEachPtr' declared void
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:17: error: 'vector' is not a member of 'std'
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:56: error: expected primary-expression before '>' token
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:60: error: 'container' was not declared in this scope
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:71: error: expected primary-expression before 'void'
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:131: error: expected primary-expression before '&&' token
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:166:134: error: expected primary-expression before '...' token
void ForEachPtr(std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:171:28: error: 'vector' in namespace 'std' does not name a template type
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:171:34: error: expected ',' or '...' before '<' token
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp: In function 'void ForEachPtr(int)':
prog.cpp:173:55: error: 'container' was not declared in this scope
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
^
prog.cpp:173:66: error: 'member_func' was not declared in this scope
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
^
prog.cpp:173:103: error: 'args' was not declared in this scope
priv_ForEachPtr_impl_forSharedOrUniquePtrsMemberFunc(container, member_func, std::forward<ParamArgs>(args)...);
^
prog.cpp: At global scope:
prog.cpp:180:22: error: variable or field 'ForEachPtr' declared void
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:17: error: 'vector' is not a member of 'std'
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:56: error: expected primary-expression before '>' token
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:60: error: 'container' was not declared in this scope
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:71: error: expected primary-expression before 'void'
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:131: error: expected primary-expression before '&&' token
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:180:134: error: expected primary-expression before '...' token
void ForEachPtr(std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:185:28: error: 'vector' in namespace 'std' does not name a template type
void ForEachPtr(const std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:185:34: error: expected ',' or '...' before '<' token
void ForEachPtr(const std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:185:6: error: redefinition of 'template<class ElementType, class ... MembFuncArgs, class ... ParamArgs> void ForEachPtr(int)'
void ForEachPtr(const std::vector<std::shared_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:171:6: note: 'template<class ElementType, class ... MembFuncArgs, class ... ParamArgs> void ForEachPtr(int)' previously declared here
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:194:22: error: variable or field 'ForEachPtr' declared void
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:17: error: 'vector' is not a member of 'std'
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:54: error: expected primary-expression before '>' token
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:58: error: 'container' was not declared in this scope
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:69: error: expected primary-expression before 'void'
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:129: error: expected primary-expression before '&&' token
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:194:132: error: expected primary-expression before '...' token
void ForEachPtr(std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...), ParamArgs&& ...args)
^
prog.cpp:199:28: error: 'vector' in namespace 'std' does not name a template type
void ForEachPtr(const std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:199:34: error: expected ',' or '...' before '<' token
void ForEachPtr(const std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:199:6: error: redefinition of 'template<class ElementType, class ... MembFuncArgs, class ... ParamArgs> void ForEachPtr(int)'
void ForEachPtr(const std::vector<std::weak_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^
prog.cpp:171:6: note: 'template<class ElementType, class ... MembFuncArgs, class ... ParamArgs> void ForEachPtr(int)' previously declared here
void ForEachPtr(const std::vector<std::unique_ptr<ElementType>> &container, void (ElementType::*member_func)(MembFuncArgs...) const, ParamArgs&& ...args)
^