// 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 #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 callback = BindBlock(^{return 42;}); // // Callback 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 R RunBlock(base::mac::ScopedBlock 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 base::Callback BindBlock(R(^block)(Args...)) { return base::Bind( &base::internal::RunBlock, base::mac::ScopedBlock( base::mac::internal::ScopedBlockTraits::Retain( block))); } #else // Construct a callback from an objective-C block with up to six arguments (see // note above). template base::Callback BindBlockArc(R (^block)(Args...)) { return base::Bind(&base::internal::RunBlock, base::mac::ScopedBlock(block)); } #endif } // namespace base #endif // BASE_MAC_BIND_OBJC_BLOCK_H_