Skip to content

Latest commit

 

History

History
288 lines (234 loc) · 11.3 KB

dispatching-events-from-your-extension.md

File metadata and controls

288 lines (234 loc) · 11.3 KB

Dispatching events from your extension

Events can be used by extensions in the following scenarios:

  • Triggering actions in the Experience Platform SDKs. Events are used by the extensions to signal when certain actions should take place, for example, to send an Adobe Analytics hit. Extensions can send the same types of events that the Experience Platform SDKs would send internally to trigger these actions.
  • Triggering actions in another extension. You may have multiple extensions in your application, and some of these extensions might have their own events defined that trigger actions.

Create an event for dispatching

The ACPExtensionEvent (iOS) / Event (Android) class contains the event that is used by the internal Event Hub. You can construct an event after the Experience Platform SDKs has been initialized.

{% hint style="info" %} This event construction is usually completed in an event listener. {% endhint %}

When constructing an event, errors can be returned for the following reasons:

  • The data that was passed cannot be converted to a JSON representation.
  • The type or source has been deprecated by Adobe.
  • The type or source is not known. For example, the com.adobe.eventType prefix is used, but the type is not known.

In the example below, a custom event called MyCustomEvent is created with custom types and sources. The internal modules respond to custom events because they are not listening for it. You can add an event listener for this type and source to verify that it is working.

{% tabs %} {% tab title="Android" %}

Android

To create events in Android, you must first import com.adobe.marketing.mobile.Event:

import com.adobe.marketing.mobile.Event;

...
List<Integer> pageOrder = new ArrayList<Integer>();
pageOrder.add(1);
pageOrder.add(2);
pageOrder.add(3);
Map<String, Object> eventData = new HashMap<String, Object>();
eventData.put("id", "clickid53");
eventData.put("localTemp", 72.6);
eventData.put("pageClicks", 23);
eventData.put("pageOrder", pageOrder);
Event newEvent = new Event.Builder("MyCustomEvent",             
                                   "com.myCompany.eventType.custom",
                                   "com.myCompany.eventSource.custom")
                                   .setEventData(eventData).build();

{% endtab %}

{% tab title="Objective-C" %}

iOS

To create events in iOS, you must first import ACPExtensionEvent.h from the ACPCore framework:

#import "ACPExtensionEvent.h"
...
NSError* error = nil;
NSDictionary* eventData = @{
            @"id" : @"clickid53",
            @"localTemp": @72.6,
            @"pageClicks" : @23,
            @"pageOrder" : @[@1,@2,@3,@4]
};
ACPExtensionEvent* newEvent = [ACPExtensionEvent extensionEventWithName:@"MyCustomEvent"          type:@"com.myCompany.eventType.custom"
        source:@"com.myCompany.eventSource.custom" 
        data:eventData
        error:&error];
if (error) {
    NSLog(@"Error constructing new event %@:%ld", [error domain], [error code]);
}

{% endtab %} {% endtabs %}

Dispatch your event

After creating your event, dispatch it by using the ACPCore (iOS) / MobileCore (Android) dispatchEvent method.

You may need to dispatch an event from one of your APIs or application methods to trigger an internal flow in your extension or another Adobe extension.

{% tabs %} {% tab title="Android" %}

Android

public void loginButtonClicked() {
    // construct the event to dispatch 
    Map<String, String> contextData = new HashMap<String, String>();
    contextData.put("username", "example@mail.com");
    contextData.put("dayofweek", "Monday");
    Map<String, Object> analyticsAction = new HashMap<String, Object>();
    analyticsAction.put("action", "clickLogin");
    analyticsAction.put("contextdata", contextData);
    Event analyticsEvent = new Event.Builder("Login click event",
                "com.adobe.eventType.generic.track",
                "com.adobe.eventSource.requestContent")
                .setEventData(analyticsAction).build();

    // dispatch the analytics event
    ExtensionErrorCallback<ExtensionError> errorCallback = new ExtensionErrorCallback<ExtensionError>() {
        @Override
        public void error(final ExtensionError extensionError) {
            Log.e("loginButtonClicked", String.format("An error occurred while dispatching                 event %d %s", extensionError.getErrorCode(), extensionError.getErrorName()));
        }
    };
    MobileCore.dispatchEvent(analyticsEvent, errorCallback);
}

{% endtab %}

{% tab title="Objective-C" %}

iOS

#import "ACPExtensionEvent.h"

- (IBAction) loginButtonClicked:(id)sender {
    // construct the event to dispatch (see above)
    ACPExtensionEvent* newEvent = ...;

    // dispatch the event
    NSError* error = nil;
    if (![ACPCore dispatchEvent:newEvent error:&error] && error) {
        NSLog(@"Error dispatching event %@:%ld", [error domain], [error code]);
    };
}

{% endtab %} {% endtabs %}

Dispatch paired events

In order to be notified when an event is being dispatched in the response of an ACPExtensionEvent (iOS) / Event (Android), use the dispatchEventWithResponseCallback API from ACPCore (iOS) / MobileCore (Android) with a non-null callback. The paired response event is sent by using the dispatchResponseEvent API.

{% hint style="info" %} Paired events are usually used for set/get operations where you need to be notified about a response event outside your extension code. {% endhint %}

The following example shows how to implement paired events:

{% tabs %} {% tab title="Android" %}

Android

You can have this code in an application's Activity or in one of your extension's API classes:

// how to dispatch a paired event with an associated response callback
ExtensionErrorCallback<ExtensionError> errorCallback = new         ExtensionErrorCallback<ExtensionError>() {
    @Override
    public void error(final ExtensionError errorCode) {
        Log.w("dispatchEventCallback", String.format("An error occurred when dispatching event, %s", errorCode.getErrorName()));
    }
};

AdobeCallback<Event> responseEventCallback = new AdobeCallback<Event>() {
    @Override
    public void call(final Event value) {
        Log.d("dispatchEventCallback", String.format("Response event received, type %s and source %s",
                                                     value.getType(), value.getSource()));
    }
};
Event event = new Event.Builder("dispatchEventWithResponseCallback", "com.myCompany.eventType.custom", "com.myCompany.eventSource.request").build();
MobileCore.dispatchEventWithResponseCallback(event, responseEventCallback, errorCallback);
...

Register a listener for this event type and source in MyExtension.java:

// register a listener for a the request event type and source
protected MyExtension(ExtensionApi extensionApi) {
        super(extensionApi);

        getApi().registerEventListener("com.myCompany.eventType.custom", "com.myCompany.eventSource.request", MyListener.class, null);
}
...

Dispatch a response event when the request is received in the hear method of your listener in MyListener.java:

public class MyListener extends ExtensionListener {

    protected MyListener(final ExtensionApi extension, final String type, final String source) {
        super(extension, type, source);
    }

    @Override
    public void hear(Event event) {
        ExtensionErrorCallback<ExtensionError> errorCallback = new ExtensionErrorCallback<ExtensionError>() {
            @Override
            public void error(final ExtensionError errorCode) {
                Log.w(getParentExtension().getName(), String.format("An error occurred when dispatching event, %s",
                        errorCode.getErrorName()));
            }
        };
        Event responseEvent = new Event.Builder("response event", "com.myCompany.eventType.custom", "com.myCompany.eventSource.response").build();

        // sending a paired response event for the request event
        MobileCore.dispatchResponseEvent(responseEvent, event, errorCallback);
    }

    @Override
    protected MyExtension getParentExtension() {
        return (MyExtension)super.getParentExtension();
    }

    ...
}

{% endtab %}

{% tab title="Objective-C" %}

iOS

You can have this code in a ViewController class or in one of your extensions public API classes:

// how to dispatch a paired event with an associated response callback
NSError *eventError = nil;
ACPExtensionEvent *event = [ACPExtensionEvent 
extensionEventWithName:@"Dispatch Request Event with Callback"
                  type:@"com.myCompany.eventType.custom"
                source:@"com.myCompany.eventSource.request"
                  data:@{}
                 error:&eventError];

if (!event && eventError) {
    NSLog(@"An error occurred while constructing event '%@': %ld", event.eventName, [eventError code]);
    return;
}

// dispatch the event and handle the callback
NSError *dispatchError = nil;
if ([ACPCore dispatchEventWithResponseCallback:event responseCallback:^(ACPExtensionEvent * _Nonnull responseEvent) {
    NSLog(@"Response event received, with type %@, source %@ and data %@", responseEvent.eventType, responseEvent.eventSource, responseEvent.eventData);
        // process event data
} error:&dispatchError]) {
    NSLog(@"Dispatched a paired request event '%@'", event.eventName);
} else if (dispatchError) {
    NSLog(@"An error occurred dispatching event '%@': %ld", event.eventName, [dispatchError code]);
}
...

Register a listener for this event type and source in MyExtension.m:

// register a listener for a the request event type and source
- (instancetype) init {
    if (self = [super init]) {
        NSError *error = nil;
        if ([self.api registerListener:[MyExtensionListener class]
                             eventType:@"com.myCompany.eventType.custom"
                           eventSource:@"com.myCompany.eventSource.request"
                                 error:&error]) {
               NSLog(@"MyExtensionListener successfully registered for request events");
         } else if (error) {
            NSLog(@"There was an error registering MyExtensionListener: %ld", [error code]);
         }
...
}

Dispatch a response event when the request is received in the hear method of your Listener in MyListener.m:

- (void) hear: (nonnull ACPExtensionEvent*) event {
    NSError *dispatchError = nil;
    NSDictionary* responseData = @{@"responsekey": @"responsevalue"};
    ACPExtensionEvent* responseEvent = [ACPExtensionEvent 
    extensionEventWithName:@"Paired Response Event"                                                                                             type:@"com.myCompany.eventType.custom"                                                                                             source:@"com.myCompany.eventSource.response"
                      data:responseData
                     error:&dispatchError];

    // sending a paired response event for the request event
    if ([ACPCore dispatchResponseEvent:responseEvent requestEvent:event error:&dispatchError]) {
        NSLog(@"Dispatched response event with data '%@'", responseData);
    } else if (dispatchError) {
        NSLog(@"An error occurred dispatching response event : %ld", [dispatchError code]);
    }
}

{% endtab %} {% endtabs %}