| // 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 BASE_MAC_BIND_OBJC_BLOCK_H_ | 
 | #define BASE_MAC_BIND_OBJC_BLOCK_H_ | 
 |  | 
 | #include <Block.h> | 
 |  | 
 | #include "base/bind.h" | 
 | #include "base/callback_forward.h" | 
 | #include "base/compiler_specific.h" | 
 | #include "base/mac/scoped_block.h" | 
 |  | 
 | // BindBlock builds a callback from an Objective-C block. Example usages: | 
 | // | 
 | // Closure closure = BindBlock(^{DoSomething();}); | 
 | // | 
 | // Callback<int(void)> callback = BindBlock(^{return 42;}); | 
 | // | 
 | // Callback<void(const std::string&, const std::string&)> callback = | 
 | //     BindBlock(^(const std::string& arg0, const std::string& arg1) { | 
 | //         ... | 
 | //     }); | 
 | // | 
 | // These variadic templates will accommodate any number of arguments, however | 
 | // the underlying templates in bind_internal.h and callback.h are limited to | 
 | // seven total arguments, and the bound block itself is used as one of these | 
 | // arguments, so functionally the templates are limited to binding blocks with | 
 | // zero through six arguments. | 
 | // | 
 | // For code compiled with ARC (automatic reference counting), use BindBlockArc. | 
 | // This is because the method has a different implementation (to avoid over- | 
 | // retaining the block) and need to have a different name not to break the ODR | 
 | // (one definition rule). Another subtle difference is that the implementation | 
 | // will call a different version of ScopedBlock constructor thus the linker must | 
 | // not merge both functions. | 
 |  | 
 | namespace base { | 
 |  | 
 | namespace internal { | 
 |  | 
 | // Helper function to run the block contained in the parameter. | 
 | template<typename R, typename... Args> | 
 | R RunBlock(base::mac::ScopedBlock<R(^)(Args...)> block, Args... args) { | 
 |   R(^extracted_block)(Args...) = block.get(); | 
 |   return extracted_block(args...); | 
 | } | 
 |  | 
 | }  // namespace internal | 
 |  | 
 | #if !defined(__has_feature) || !__has_feature(objc_arc) | 
 |  | 
 | // Construct a callback from an objective-C block with up to six arguments (see | 
 | // note above). | 
 | template<typename R, typename... Args> | 
 | base::Callback<R(Args...)> BindBlock(R(^block)(Args...)) { | 
 |   return base::Bind( | 
 |       &base::internal::RunBlock<R, Args...>, | 
 |       base::mac::ScopedBlock<R (^)(Args...)>( | 
 |           base::mac::internal::ScopedBlockTraits<R (^)(Args...)>::Retain( | 
 |               block))); | 
 | } | 
 |  | 
 | #else | 
 |  | 
 | // Construct a callback from an objective-C block with up to six arguments (see | 
 | // note above). | 
 | template <typename R, typename... Args> | 
 | base::Callback<R(Args...)> BindBlockArc(R (^block)(Args...)) { | 
 |   return base::Bind(&base::internal::RunBlock<R, Args...>, | 
 |                     base::mac::ScopedBlock<R (^)(Args...)>(block)); | 
 | } | 
 |  | 
 | #endif | 
 |  | 
 | }  // namespace base | 
 |  | 
 | #endif  // BASE_MAC_BIND_OBJC_BLOCK_H_ |