macos/foundation: number, dict get value

This commit is contained in:
Mitchell Hashimoto
2022-10-01 15:49:05 -07:00
parent 3eeef74315
commit 7eb466959a
4 changed files with 104 additions and 27 deletions

View File

@ -2,6 +2,7 @@ pub const c = @import("foundation/c.zig");
pub usingnamespace @import("foundation/array.zig"); pub usingnamespace @import("foundation/array.zig");
pub usingnamespace @import("foundation/base.zig"); pub usingnamespace @import("foundation/base.zig");
pub usingnamespace @import("foundation/dictionary.zig"); pub usingnamespace @import("foundation/dictionary.zig");
pub usingnamespace @import("foundation/number.zig");
pub usingnamespace @import("foundation/string.zig"); pub usingnamespace @import("foundation/string.zig");
pub usingnamespace @import("foundation/type.zig"); pub usingnamespace @import("foundation/type.zig");
pub usingnamespace @import("foundation/url.zig"); pub usingnamespace @import("foundation/url.zig");

View File

@ -0,0 +1,3 @@
pub usingnamespace @cImport({
@cInclude("CoreFoundation/CoreFoundation.h");
});

View File

@ -2,11 +2,12 @@ const std = @import("std");
const assert = std.debug.assert; const assert = std.debug.assert;
const Allocator = std.mem.Allocator; const Allocator = std.mem.Allocator;
const foundation = @import("../foundation.zig"); const foundation = @import("../foundation.zig");
const c = @import("c.zig");
pub const Dictionary = opaque { pub const Dictionary = opaque {
pub fn create( pub fn create(
keys: ?[]*const anyopaque, keys: ?[]?*const anyopaque,
values: ?[]*const anyopaque, values: ?[]?*const anyopaque,
) Allocator.Error!*Dictionary { ) Allocator.Error!*Dictionary {
if (keys != null or values != null) { if (keys != null or values != null) {
assert(keys != null); assert(keys != null);
@ -14,14 +15,14 @@ pub const Dictionary = opaque {
assert(keys.?.len == values.?.len); assert(keys.?.len == values.?.len);
} }
return CFDictionaryCreate( return @intToPtr(?*Dictionary, @ptrToInt(c.CFDictionaryCreate(
null, null,
if (keys) |slice| slice.ptr else null, @ptrCast([*c]?*const anyopaque, if (keys) |slice| slice.ptr else null),
if (values) |slice| slice.ptr else null, @ptrCast([*c]?*const anyopaque, if (values) |slice| slice.ptr else null),
if (keys) |slice| slice.len else 0, @intCast(c.CFIndex, if (keys) |slice| slice.len else 0),
&kCFTypeDictionaryKeyCallBacks, &c.kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks, &c.kCFTypeDictionaryValueCallBacks,
) orelse Allocator.Error.OutOfMemory; ))) orelse Allocator.Error.OutOfMemory;
} }
pub fn release(self: *Dictionary) void { pub fn release(self: *Dictionary) void {
@ -29,36 +30,29 @@ pub const Dictionary = opaque {
} }
pub fn getCount(self: *Dictionary) usize { pub fn getCount(self: *Dictionary) usize {
return CFDictionaryGetCount(self); return @intCast(usize, c.CFDictionaryGetCount(@ptrCast(c.CFDictionaryRef, self)));
} }
pub extern "c" fn CFDictionaryCreate( pub fn getValue(self: *Dictionary, comptime V: type, key: ?*const anyopaque) ?*V {
allocator: ?*anyopaque, return @intToPtr(?*V, @ptrToInt(c.CFDictionaryGetValue(
keys: ?[*]*const anyopaque, @ptrCast(c.CFDictionaryRef, self),
values: ?[*]*const anyopaque, key,
num_values: usize, )));
key_callbacks: *const anyopaque, }
value_callbacks: *const anyopaque,
) ?*Dictionary;
pub extern "c" fn CFDictionaryGetCount(*Dictionary) usize;
extern "c" var kCFTypeDictionaryKeyCallBacks: anyopaque;
extern "c" var kCFTypeDictionaryValueCallBacks: anyopaque;
}; };
// Just used for a test
extern "c" var kCFURLIsPurgeableKey: *const anyopaque;
test "dictionary" { test "dictionary" {
const testing = std.testing; const testing = std.testing;
const str = try foundation.String.createWithBytes("hello", .unicode, false); const str = try foundation.String.createWithBytes("hello", .unicode, false);
defer str.release(); defer str.release();
var keys = [_]*const anyopaque{kCFURLIsPurgeableKey}; var keys = [_]?*const anyopaque{c.kCFURLIsPurgeableKey};
var values = [_]*const anyopaque{str}; var values = [_]?*const anyopaque{str};
const dict = try Dictionary.create(&keys, &values); const dict = try Dictionary.create(&keys, &values);
defer dict.release(); defer dict.release();
try testing.expectEqual(@as(usize, 1), dict.getCount()); try testing.expectEqual(@as(usize, 1), dict.getCount());
try testing.expect(dict.getValue(foundation.String, c.kCFURLIsPurgeableKey) != null);
try testing.expect(dict.getValue(foundation.String, c.kCFURLIsVolumeKey) == null);
} }

View File

@ -0,0 +1,79 @@
const std = @import("std");
const Allocator = std.mem.Allocator;
const foundation = @import("../foundation.zig");
const c = @import("c.zig");
pub const Number = opaque {
pub fn create(
comptime type_: NumberType,
value: *const type_.ValueType(),
) Allocator.Error!*Number {
return @intToPtr(?*Number, @ptrToInt(c.CFNumberCreate(
null,
@enumToInt(type_),
value,
))) orelse Allocator.Error.OutOfMemory;
}
pub fn getValue(self: *Number, comptime t: NumberType, ptr: *t.ValueType()) bool {
return c.CFNumberGetValue(
@ptrCast(c.CFNumberRef, self),
@enumToInt(t),
ptr,
) == 1;
}
pub fn release(self: *Number) void {
c.CFRelease(self);
}
};
pub const NumberType = enum(c.CFNumberType) {
sint8 = c.kCFNumberSInt8Type,
sint16 = c.kCFNumberSInt16Type,
sint32 = c.kCFNumberSInt32Type,
sint64 = c.kCFNumberSInt64Type,
float32 = c.kCFNumberFloat32Type,
float64 = c.kCFNumberFloat64Type,
char = c.kCFNumberCharType,
short = c.kCFNumberShortType,
int = c.kCFNumberIntType,
long = c.kCFNumberLongType,
long_long = c.kCFNumberLongLongType,
float = c.kCFNumberFloatType,
double = c.kCFNumberDoubleType,
cf_index = c.kCFNumberCFIndexType,
ns_integer = c.kCFNumberNSIntegerType,
cg_float = c.kCFNumberCGFloatType,
pub fn ValueType(self: NumberType) type {
return switch (self) {
.sint8 => i8,
.sint16 => i16,
.sint32 => i32,
.sint64 => i64,
.float32 => f32,
.float64 => f64,
.char => u8,
.short => c_short,
.int => c_int,
.long => c_long,
.long_long => c_longlong,
.float => f32,
.double => f64,
else => unreachable, // TODO
};
}
};
test {
const testing = std.testing;
const inner: i8 = 42;
const v = try Number.create(.sint8, &inner);
defer v.release();
var result: i8 = undefined;
try testing.expect(v.getValue(.sint8, &result));
try testing.expectEqual(result, inner);
}