blob: d186f3d8c0e9517b4990750da9b721f09bca2814 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROME_BROWSER_GOOGLE_APIS_TASK_UTIL_H_
#define CHROME_BROWSER_GOOGLE_APIS_TASK_UTIL_H_
#include "base/bind.h"
#include "base/message_loop/message_loop_proxy.h"
namespace google_apis {
// Runs task on the thread to which |relay_proxy| belongs.
void RunTaskOnThread(scoped_refptr<base::MessageLoopProxy> relay_proxy,
const base::Closure& task);
namespace internal {
// Implementation of the composed callback, whose signature is |Sig|.
template<typename Sig> struct ComposedCallback;
// ComposedCallback with no argument.
template<>
struct ComposedCallback<void()> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Closure& callback) {
runner.Run(callback);
}
};
// ComposedCallback with one argument.
template<typename T1>
struct ComposedCallback<void(T1)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1)>& callback,
T1 arg1) {
runner.Run(base::Bind(callback, arg1));
}
};
// ComposedCallback with two arguments.
template<typename T1, typename T2>
struct ComposedCallback<void(T1, T2)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, T2)>& callback,
T1 arg1, T2 arg2) {
runner.Run(base::Bind(callback, arg1, arg2));
}
};
// ComposedCallback with two arguments, and the last one is scoped_ptr.
template<typename T1, typename T2, typename D2>
struct ComposedCallback<void(T1, scoped_ptr<T2, D2>)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, scoped_ptr<T2, D2>)>& callback,
T1 arg1, scoped_ptr<T2, D2> arg2) {
runner.Run(base::Bind(callback, arg1, base::Passed(&arg2)));
}
};
// ComposedCallback with three arguments.
template<typename T1, typename T2, typename T3>
struct ComposedCallback<void(T1, T2, T3)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, T2, T3)>& callback,
T1 arg1, T2 arg2, T3 arg3) {
runner.Run(base::Bind(callback, arg1, arg2, arg3));
}
};
// ComposedCallback with three arguments, and the last one is scoped_ptr.
template<typename T1, typename T2, typename T3, typename D3>
struct ComposedCallback<void(T1, T2, scoped_ptr<T3, D3>)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, T2, scoped_ptr<T3, D3>)>& callback,
T1 arg1, T2 arg2, scoped_ptr<T3, D3> arg3) {
runner.Run(base::Bind(callback, arg1, arg2, base::Passed(&arg3)));
}
};
// ComposedCallback with four arguments.
template<typename T1, typename T2, typename T3, typename T4>
struct ComposedCallback<void(T1, T2, T3, T4)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, T2, T3, T4)>& callback,
T1 arg1, T2 arg2, T3 arg3, T4 arg4) {
runner.Run(base::Bind(callback, arg1, arg2, arg3, arg4));
}
};
// ComposedCallback with four arguments, and the second one is scoped_ptr.
template<typename T1, typename T2, typename D2, typename T3, typename T4>
struct ComposedCallback<void(T1, scoped_ptr<T2, D2>, T3, T4)> {
static void Run(
const base::Callback<void(const base::Closure&)>& runner,
const base::Callback<void(T1, scoped_ptr<T2, D2>, T3, T4)>& callback,
T1 arg1, scoped_ptr<T2, D2> arg2, T3 arg3, T4 arg4) {
runner.Run(base::Bind(callback, arg1, base::Passed(&arg2), arg3, arg4));
}
};
// GetDefaultValue returns a value constructed by the default constructor.
template<typename T> struct DefaultValueCreator {
static T GetDefaultValue() { return T(); }
};
template<typename T> struct DefaultValueCreator<const T&> {
static T GetDefaultValue() { return T(); }
};
// Helper of CreateErrorRunCallback implementation.
// Provides:
// - ResultType; the type of the Callback which should be returned by
// CreateErrorRunCallback.
// - Run(): a static function which takes the original |callback| and |error|,
// and runs the |callback|.Run() with the error code and default values
// for remaining arguments.
template<typename CallbackType> struct CreateErrorRunCallbackHelper;
// CreateErrorRunCallback with two arguments.
template<typename ErrorType, typename P1>
struct CreateErrorRunCallbackHelper<void(ErrorType, P1)> {
typedef base::Callback<void(ErrorType)> ResultType;
static void Run(
const base::Callback<void(ErrorType, P1)>& callback, ErrorType error) {
callback.Run(error, DefaultValueCreator<P1>::GetDefaultValue());
}
};
} // namespace internal
// Returns callback that takes arguments (arg1, arg2, ...), create a closure
// by binding them to |callback|, and runs |runner| with the closure.
// I.e. the returned callback works as follows:
// runner.Run(Bind(callback, arg1, arg2, ...))
template<typename CallbackType>
CallbackType CreateComposedCallback(
const base::Callback<void(const base::Closure&)>& runner,
const CallbackType& callback) {
DCHECK(!runner.is_null());
DCHECK(!callback.is_null());
return base::Bind(
&internal::ComposedCallback<typename CallbackType::RunType>::Run,
runner, callback);
}
// Returns callback which runs the given |callback| on the current thread.
template<typename CallbackType>
CallbackType CreateRelayCallback(const CallbackType& callback) {
return CreateComposedCallback(
base::Bind(&RunTaskOnThread, base::MessageLoopProxy::current()),
callback);
}
// Returns a callback with the tail parameter bound to its default value.
// In other words, returned_callback.Run(error) runs callback.Run(error, T()).
template<typename CallbackType>
typename internal::CreateErrorRunCallbackHelper<CallbackType>::ResultType
CreateErrorRunCallback(const base::Callback<CallbackType>& callback) {
return base::Bind(
&internal::CreateErrorRunCallbackHelper<CallbackType>::Run, callback);
}
} // namespace google_apis
#endif // CHROME_BROWSER_GOOGLE_APIS_TASK_UTIL_H_