不使用桥接头文件如何访问私有 UIKit 函数

11
考虑私有C函数_UICreateScreenUIImage,它返回当前设备屏幕的UIImage快照:
OBJC_EXTERN UIImage *_UICreateScreenUIImage(void) NS_RETURNS_RETAINED;

我可以把这个放在一个桥接头文件中,然后在 Swift 中像这样访问:

MyApp-Bridging-Header.h

@import UIKit;
UIImage *_UICreateScreenUIImage(void) NS_RETURNS_RETAINED;

MyClass.swift

let image = _UICreateScreenUIImage()
print(image) // <UIImage: 0x7fc4ba6081c0>, {375, 667}
有没有一种方法可以在不使用桥接头文件的情况下,在纯Swift中访问_UICreateScreenUIImage
最初的想法是在UIImage的扩展上创建一个函数,但扩展要求我在扩展内部声明该函数的主体:
extension UIImage {
    public func _UICreateScreenUIImage(_: Void) -> UIImage // "Expected '{' in body of function declaration"
}

无论如何,此实现存在缺陷,因为_UICreateScreenUIImage不是UIImage的方法。

在纯Swift中暴露和访问此方法是否可能?


人们似乎将我的问题与“我如何截取屏幕截图?”混淆了。那不是我在问的。我正在询问如何访问像UIImage *_UICreateScreenUIImage(void);这样的方法。它可以是任何私有方法,比如+(UIImage *)_deviceSpecificImageNamed:(NSString *)name inBundle:(NSBundle *)bundle;或者+(UIImage *)_pu_PhotosUIImageNamed:(NSString *)name;

1个回答

18

这比您想象的要容易得多:

@asmname("_UICreateScreenUIImage")
func _UICreateScreenUIImage() -> UIImage

// That's it – go ahead and call it:
_UICreateScreenUIImage()

事实上,在2.3版本中,@asmname已经被更改为@_silgen_name,所以请做好相应的调整准备:刚好

@_silgen_name("_UICreateScreenUIImage")
func _UICreateScreenUIImage() -> UIImage
据我所知,@_silgen_name 不能解决 Objective-C 方法。对此,有更加强大的 Objective-C 运行时 API:
let invokeImageNamed: (String, NSTimeInterval) -> UIImage? = {
    // The Objective-C selector for the method.
    let selector: Selector = "animatedImageNamed:duration:"
    guard case let method = class_getClassMethod(UIImage.self, selector)
        where method != nil else { fatalError("Failed to look up \(selector)") }

    // Recreation of the method's implementation function.
    typealias Prototype = @convention(c) (AnyClass, Selector, NSString, NSTimeInterval) -> UIImage?
    let opaqueIMP = method_getImplementation(method)
    let function = unsafeBitCast(opaqueIMP, Prototype.self)

    // Capture the implemenation data in a closure that can be invoked at any time.
    return { name, interval in function(UIImage.self, selector, name, interval) }
}()

extension UIImage {
    // Convenience method for calling the closure from the class.
    class func imageNamed(name: String, interval: NSTimeInterval) -> UIImage? {
        return invokeImageNamed(name, interval)
    }
}

UIImage.imageNamed("test", interval: 0)

关于处理NS_RETURNS_RETAINED,这不会为您生成。 相反,您可以使用Unmanaged作为返回类型,并将其包装在方便的函数中:

@_silgen_name("_UICreateScreenUIImage")
func _UICreateScreenUIImage() -> Unmanaged<UIImage>
func UICreateScreenUIImage() -> UIImage {
    return _UICreateScreenUIImage().takeRetainedValue()
}

1
这个有效!太棒了!asmname_silgen_name在哪里有记录?这是Swift中的extern C等价物吗? - JAL
2
据我所知,这并没有记录(除了LLVM的内部,我相信)。术语“asmname”表明它确实等同于“extern”,它只是指示汇编器在其他地方寻找符号。而“_silgen_name”则更具体地意味着它在SIL生成期间产生“builtin ""()`”声明(这些声明又类似于“extern”由汇编器解析)。 - Sam R.
2
非常感谢!我期望将其标记为已接受并授予您赏金。另一个后续问题:如果我想访问UIImage上的私有类方法本身,例如:+(UIImage *)_pu_PhotosUIImageNamed:(NSString *)name;?那是否遵循相同的模式? - JAL
1
另外,我如何将函数标记为“NS_RETURNS_RETAINED”,以便无需手动管理内存? - JAL
1
很不幸,它们并不完全相同。我已经编辑了我的答案,涵盖了NS_RETURNS_RETAINED问题。 - Sam R.
1
感谢您的跟进,非常感谢您的出色回答! - JAL

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接