Skip to content

Commit 1acdf9b

Browse files
authored
Allow use of @static within @objcproperties. (#48)
1 parent 16c257e commit 1acdf9b

File tree

2 files changed

+83
-12
lines changed

2 files changed

+83
-12
lines changed

src/syntax.jl

+28-12
Original file line numberDiff line numberDiff line change
@@ -431,27 +431,43 @@ macro objcproperties(typ, ex)
431431
read_properties = Dict{Symbol,Expr}()
432432
write_properties = Dict{Symbol,Expr}()
433433

434-
for arg in ex.args
435-
isa(arg, LineNumberNode) && continue
436-
Meta.isexpr(arg, :macrocall) || propertyerror("invalid property declaration $arg")
434+
# collect property declarations
435+
properties = []
436+
function process_property(ex)
437+
isa(ex, LineNumberNode) && return
438+
Meta.isexpr(ex, :macrocall) || propertyerror("invalid property declaration $ex")
437439

438440
# split the contained macrocall into its parts
439-
cmd = arg.args[1]
441+
cmd = ex.args[1]
442+
args = []
440443
kwargs = Dict()
441-
positionals = []
442-
for arg in arg.args[2:end]
444+
for arg in ex.args[2:end]
443445
isa(arg, LineNumberNode) && continue
444446
if isa(arg, Expr) && arg.head == :(=)
445447
kwargs[arg.args[1]] = arg.args[2]
446448
else
447-
push!(positionals, arg)
449+
push!(args, arg)
448450
end
449451
end
450452

453+
# if we're dealing with `@static`, so recurse into the block
454+
# TODO: liberally support all unknown macros?
455+
if cmd == Symbol("@static")
456+
ex = macroexpand(__module__, ex; recursive=false)
457+
if ex !== nothing
458+
process_property.(ex.args)
459+
end
460+
else
461+
push!(properties, (; cmd, args, kwargs))
462+
end
463+
end
464+
process_property.(ex.args)
465+
466+
for (cmd, args, kwargs) in properties
451467
# there should only be a single positional argument,
452468
# containing the property name (and optionally its type)
453-
length(positionals) >= 1 || propertyerror("$cmd requires a positional argument")
454-
property_arg = popfirst!(positionals)
469+
length(args) >= 1 || propertyerror("$cmd requires a positional argument")
470+
property_arg = popfirst!(args)
455471
if property_arg isa Symbol
456472
property = property_arg
457473
srcTyp = nothing
@@ -509,14 +525,14 @@ macro objcproperties(typ, ex)
509525
end
510526
elseif cmd == Symbol("@getproperty")
511527
haskey(read_properties, property) && propertyerror("duplicate property $property")
512-
function_arg = popfirst!(positionals)
528+
function_arg = popfirst!(args)
513529
read_properties[property] = quote
514530
f = $(esc(function_arg))
515531
f(object)
516532
end
517533
elseif cmd == Symbol("@setproperty!")
518534
haskey(write_properties, property) && propertyerror("duplicate property $property")
519-
function_arg = popfirst!(positionals)
535+
function_arg = popfirst!(args)
520536
write_properties[property] = quote
521537
f = $(esc(function_arg))
522538
f(object, value)
@@ -525,7 +541,7 @@ macro objcproperties(typ, ex)
525541
propertyerror("unrecognized property declaration $cmd")
526542
end
527543

528-
isempty(positionals) || propertyerror("too many positional arguments")
544+
isempty(args) || propertyerror("too many positional arguments")
529545
end
530546

531547
# generate Base.propertynames definition

test/runtests.jl

+55
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,61 @@ end
3636
@test_throws UndefRefError TestNSString(nil)
3737
end
3838

39+
@objcproperties TestNSString begin
40+
@autoproperty length::Culong
41+
@static if true
42+
@autoproperty UTF8String::Ptr{Cchar}
43+
end
44+
@static if false
45+
@autoproperty NonExistingProperty::Cint
46+
end
47+
end
48+
@objcwrapper TestNSMutableString <: TestNSString
49+
@objcproperties TestNSMutableString begin
50+
@setproperty! string function(obj, val)
51+
@objc [obj::id{TestNSMutableString} setString:val::id{TestNSString}]::Nothing
52+
end
53+
end
54+
@objcwrapper TestNSOperationQueue <: Object
55+
@objcproperties TestNSOperationQueue begin
56+
@autoproperty name::id{TestNSString} setter=setName
57+
end
58+
@testset "@objcproperties" begin
59+
# immutable object with only read properties
60+
str1 = "foo"
61+
immut = TestNSString(@objc [NSString stringWithUTF8String:str1::Ptr{UInt8}]::id{TestNSString})
62+
63+
@test :length in propertynames(immut)
64+
@test :UTF8String in propertynames(immut)
65+
@test :NonExistingProperty propertynames(immut)
66+
67+
@test immut.length == length(str1)
68+
@test unsafe_string(immut.UTF8String) == str1
69+
70+
# mutable object with a write property
71+
str2 = "barbar"
72+
mut = TestNSMutableString(@objc [NSMutableString stringWithUTF8String:str2::Ptr{UInt8}]::id{TestNSMutableString})
73+
74+
@test :length in propertynames(mut)
75+
@test :UTF8String in propertynames(mut)
76+
@test :string in propertynames(mut)
77+
@test :NonExistingProperty propertynames(mut)
78+
79+
@test mut.length == length(str2)
80+
@test unsafe_string(mut.UTF8String) == str2
81+
82+
mut.string = immut
83+
@test mut.length == length(str1)
84+
@test unsafe_string(mut.UTF8String) == str1
85+
86+
# mutable object using @autoproperty to generate a setter
87+
queue = TestNSOperationQueue(@objc [NSOperationQueue new]::id{TestNSOperationQueue})
88+
@test queue.name isa TestNSString
89+
@test unsafe_string(queue.name.UTF8String) != str1
90+
queue.name = immut
91+
@test unsafe_string(queue.name.UTF8String) == str1
92+
end
93+
3994
@testset "@objc blocks" begin
4095
# create a dummy class we'll register our blocks with
4196
# (no need to use @objcwrapper as we're not constructing an id{BlockWrapper})

0 commit comments

Comments
 (0)