Actions

Often child components must communicate back to their parents. For example, a button component may need to signal that it has been tapped. Component actions provide a way to do this.

What are Component Actions?

CKAction<T...> is an Objective-C++ class that wraps a SEL (basically a method name in Objective-C), and a target. CKAction<T...> allows you to specify the types of the arguments that are provided to the receiving method.

Where possible, you should explicitly define the target of your action using either a component scope, or a target for non-Component targets.

An action may be sent through the send function on CKAction, which takes the sender component, and the parameters to be passed to the receiver.

For legacy reasons, we also support using CKActionSend. The CKActionSend function takes an action, a sender, and an optional opaque context object. It follows the component responder chain until it finds a component (or component controller) that responds to the given selector, then sends a message with the sender and context as parameters.

CKComponentActionSend must be called on the main thread.

Using Component Actions

Here's an example of how to handle a component action.

@implementation SampleComponent
+ (instancetype)new
{
CKComponentScope scope(self);
return [super newWithComponent:
CK::ButtonComponentBuilder()
.action({scope, @selector(someAction:event:)})
.build()];
}
- (void)someAction:(CKButtonComponent *)sender event:(UIEvent *)event
{
// Do something
}
@end
@implementation SampleOtherComponentThatDoesntCareAboutEvents
+ (instancetype)new
{
CKComponentScope scope(self);
return [super newWithComponent:
CK::ButtonComponentBuilder()
.action({scope, @selector(someAction:)})
.build()];
}
- (void)someAction:(CKButtonComponent *)sender
{
// Do something
}
@end
@implementation SampleOtherComponentThatDoesntCareAboutAnyParameters
+ (instancetype)new
{
CKComponentScope scope(self);
return [super newWithComponent:
CK::ButtonComponentBuilder()
.action({scope, @selector(someAction)})
.build()];
}
- (void)someAction
{
// We don't take any arguments in this example.
}
@end
@interface SampleControllerDelegatingComponentController : CKComponentController
/** Component actions may be implemented either on the component, or the controller for that component. */
- (void)someAction;
@end
@implementation SampleControllerDelegatingComponent
+ (instancetype)new
{
CKComponentScope scope(self);
return [super newWithComponent:
CK::ButtonComponentBuilder()
.action({scope, @selector(someAction)})
.build()];
}
@end
@implementation SampleControllerDelegatingComponentController
- (void)someAction
{
// Do something
}
@end

Component actions will only be sent up the component responder chain in a mounted component hierarchy. Trying to call send on a responder chain-based action from an unmounted component will raise an assertion. Use a scope-based component action if you need to send an action while not mounted.

CKAction is a C++ object which can lead to subtle problems when a CKAction reference (e.g. CKAction & or const CKAction &) is captured by a block. The reason? C++ references are not managed by ARC, which can result in crashes that are difficult to debug. The best way to avoid this problem is the pass actions by value, or by explicitly creating a copy for use within the block.

Why not blocks?

Blocks might seem like a more natural way to specify callbacks. Unfortunately it's far too easy to introduce retain cycles when using blocks: components hold strong references to their child components, and the child might hold a block with a strong reference back to the parent.

Hybrid Responder Chain

Component actions begin traversal of the component responder chain from the target or scoped component, or if neither are defined, at the sender of the action. They traverse upwards from there.

In general, you should avoid using the component responder chain, but for legacy reasons it still exists. Instead, you should use target- or scope-based actions, which will verify that the action is handled directly. If you use the scope-based component action, the type-checking machinery will verify that your component or controller responds to the selector with the expected param types (to the extent that Obj-C allows) at runtime.

Automatic Promotion

In order to support a progressive adoption of typed actions, we allow automatic "promotion" of component actions. By promotion, we mean you can provide a component action that takes less arguments to a component that expects more arguments. So, for instance, you can provide a CKAction<id>(scope, @selector(actionWithSender:firstParam:)) to a component that expects a CKAction<id, id>. At runtime, your method will simply not receive the additional parameters that it does not expect.

Explicit Demotion

Legacy callsites also may use "demotion", but it is disabled by default. By demotion, we mean providing a component action that expects more parameters than handled. So for instance, this would be like passing CKAction<id, id, id> to a component which expects an action of type CKAction<id>. In this case, at action-call time we would be getting less parameters than we expected. For this reason, we have forced these conversions to be explicit. You must explicitly convert your action to demote it by calling the demoted copy constructor: CKAction<id>(CKAction<id, id, id>(scope, @selector(someMethodWithSender:param1:param2:param3:))). At runtime the parameters that aren't provided by the action will be filled with zeros (so nil for object types, zerod structs or primitives).