Yes, please.
Off the top of my head, suppose you wanted to describe @IBAction
. This attribute has a pretty complex set of rules about what it supports; you might express them with something like this:
@staticAttribute struct IBAction {
// This is the initializer used for the attribute. It needs to be marked
// to distinguish it from the other members.
@staticAttribute init() {}
// The remaining members are all "shapes" of declarations that you can
// validly apply this attribute to. If the declaration "looks" like one
// of these, ignoring the names but looking at things like the declaration
// kind, parameter counts and types, result type, throwing-ness, and
// attributes, then you can apply the attribute to that declaration.
#if os(macOS)
// macOS allows only one signature, with a sender parameter and Void result.
@objc func action1(_ sender: AnyObject?) {}
#else
// iOS supports zero, one, or two parameters.
@objc func action0() {}
@objc func action1(_ sender: AnyObject?) {}
@objc func action2(_ sender: AnyObject?, _ event: AnyObject?) {}
// To support WatchKit, it also supports a bunch of variants which take
// value types. I don't remember exactly which ones WatchKit uses, but
// that's fine, this is just an example.
@objc func actionInt(_ value: Int) {}
@objc func actionDouble(_ value: Double) {}
@objc func actionBool(_ value: Bool) {}
#endif
}
This idea is at best half-baked, so it probably makes sense to defer it, but hopefully you get the idea.