|  | // 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_ |