CFDictionaryGetValueIfPresent 看起来像这样:
Boolean CFDictionaryGetValueIfPresent(CFDictionaryRef theDict, const void *key, const void **value);
在 core_foundation 中是这样的:
pub unsafe extern "C" fn CFDictionaryGetValueIfPresent(
theDict: CFDictionaryRef,
key: *const c_void,
value: *mut *const c_void
) -> Boolean
我相信关键是CFStringRef;在 C 中,有一个宏可以让这更容易(CFSTR("cstring")),这里我们必须使用稍长的形式:
let c_key = CString::new("kCGWindowOwnerName").unwrap();
let cf_key = unsafe {
CFStringCreateWithCString(std::ptr::null(), c_key.as_ptr(), kCFStringEncodingUTF8)
};
这给了我们CFStringRef,在 core_foundation 中它是这样定义的:
pub type CFStringRef = *const __CFString;
它已经是一个*const 指针,只是不是我们需要的类型。你可以改为c_void:
let key_ptr: *const c_void = unsafe { std::mem::transmute(cf_key) };
value 参数只需要一个原始双指针来存储结果。创建空指针:
let mut value: *const c_void = std::ptr::null();
并传递对其的可变引用 (&mut value) 以获取 *mut *const c_void。
您的 rust 代码中有一些错误和奇怪之处。这是一个带注释的工作示例:
use core_graphics::display::*;
use core_foundation::string::*;
use std::ffi::{ CStr, CString, c_void };
fn main() {
// CGWindowListOption is a bitmask, combine the flags with bitwise OR
const OPTIONS: CGWindowListOption = kCGWindowListOptionOnScreenOnly | kCGWindowListExcludeDesktopElements;
// No need to specify the type or use 'as'; CFArrayRef is the return type from CGWindowListCopyWindowInfo
let window_list_info = unsafe { CGWindowListCopyWindowInfo(OPTIONS, kCGNullWindowID) };
// Don't use const here, CFArrayGetCount returns CFIndex (long)
let count: i64 = unsafe { CFArrayGetCount(window_list_info) };
for i in 0..count-1 {
// Here we need the 'as', CFArrayGetValueAtIndex just returns void*
let dic_ref = unsafe { CFArrayGetValueAtIndex(window_list_info, i) as CFDictionaryRef };
// Create a CString from the key name we are interested in
let c_key = CString::new("kCGWindowOwnerName").unwrap();
// Create a CFString, needs to be released with `CFRelease`. I leave that as an exercise for the reader.
let cf_key = unsafe { CFStringCreateWithCString(std::ptr::null(), c_key.as_ptr(), kCFStringEncodingUTF8) };
// cf_key is a CFStringRef, which is a type alias to *const __CFString
// We transmute it into *const c_void since that is what CFDictionaryGetValueIfPresent wants
let key_ptr: *const c_void = unsafe { std::mem::transmute(cf_key) };
// A raw void* to hold the result
let mut value: *const c_void = std::ptr::null();
if unsafe { CFDictionaryGetValueIfPresent(dic_ref, key_ptr, &mut value) != 0 } {
// CFDictionaryGetValueIfPresent returned true; that means value must point to a CFStringRef
let cf_ref = value as core_foundation::string::CFStringRef;
// Get a pointer to a C-string buffer with the characters from the CFString
let c_ptr = unsafe { CFStringGetCStringPtr(cf_ref, kCFStringEncodingUTF8) };
// The value may be null; don't pass it to CStr::from_ptr
if c_ptr.is_null() { continue; }
// Wrap the pointer in a rust CStr so we can convert a str
let c_value = unsafe { CStr::from_ptr(c_ptr) };
println!("{}", c_value.to_str().unwrap());
}
}
}
作为参考,这里是 C 中的相同示例:
#import <Foundation/Foundation.h>
#import <CoreGraphics/CoreGraphics.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
CGWindowListOption options = kCGWindowListOptionOnScreenOnly | kCGWindowListExcludeDesktopElements;
CFArrayRef windows = CGWindowListCopyWindowInfo(options, kCGNullWindowID);
CFIndex count = CFArrayGetCount(windows);
for (int i = 0; i < count; i++)
{
CFDictionaryRef windowDict = CFArrayGetValueAtIndex(windows, i);
CFStringRef key = CFStringCreateWithCString(NULL, "kCGWindowOwnerName", kCFStringEncodingUTF8);
const void* value = nil;
if (CFDictionaryGetValueIfPresent(windowDict, key, &value) == YES)
{
const char* c_value = CFStringGetCStringPtr(value, kCFStringEncodingUTF8);
NSLog(@"window: %s", c_value);
}
CFRelease(key);
}
}
return 0;
}
免责声明我对 rust 也比较陌生,这可能不是惯用的解决方案