mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 14:26:09 +03:00
80 lines
2.7 KiB
C
80 lines
2.7 KiB
C
|
// 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_
|