Skip to content

Latest commit

 

History

History
356 lines (269 loc) · 12.2 KB

File metadata and controls

356 lines (269 loc) · 12.2 KB

Listening for events

A common use case is to add an event listener to get notifications for events. The main location to add an event listener is in the init method, although you can add listeners by using other callbacks later. You can add the logic that you want executed when an event occurs, and for which you have a listener, in the hear method of your listener class.

When handling an event in the event listener, note that the hear method should take a maximum of 100ms to execute. This means that potentially long-running operations should be pushed to another thread (for example, network or file operations), as in the following examples.

Here are some additional rules to note for event listeners:

  • You can listen to an event by registering your listener class with an event source and an event type. The Adobe Experience Platform SDKs create an instance of your listener class and retain it as long as your extension is registered.
  • When an event that you are listening for occurs, the hear method on the appropriate instance of your listener class is called.
  • You can register multiple listeners, but each listener instance only listens for one source/type pair.
  • One listener class might be used to listen for multiple events, but you need to check the details of the event you are passed on each call to the hear method.
  • All registered listeners are released when the extension is unregistered.

The event type and event source that are associated with the received event can be in lower case because the Mobile SDK compares the type and the source as case insensitive values. When you check the event type and event source of the event, we recommend that you use the ignore case string comparison.

Creating your event listener

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

Android

Create a new Java class for your listener, extend the base class ExtensionListener, and implement the required constructor and hear method. You can also override the getParentExtension method to retrieve your custom extension class instead of the base Extension class.

package com.test.company;

import com.adobe.marketing.mobile.Extension;
import com.adobe.marketing.mobile.ExtensionListener;
import com.adobe.marketing.mobile.ExtensionApi;
import com.adobe.marketing.mobile.Event;

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

    @Override
    public void hear(final Event event) {
        // run the event processing on its own executor in the parent extension class
        getParentExtension().handleConfigurationEvent(event);
    }

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

{% hint style="info" %} Create an executor in your Extension class for event processing and use it when one of your listeners is called. This way you do not have to worry about how long your event handler is taking. {% endhint %}

package com.test.company;

import android.util.Log;

import com.adobe.marketing.mobile.Event;
import com.adobe.marketing.mobile.Extension;
import com.adobe.marketing.mobile.ExtensionApi;
import com.adobe.marketing.mobile.ExtensionError;
import com.adobe.marketing.mobile.ExtensionErrorCallback;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class MyExtension extends Extension {
    private final Object executorMutex = new Object();
    private ExecutorService executor;

    public MyExtension(final ExtensionApi moduleApi) {
        super(moduleApi);

        ExtensionErrorCallback<ExtensionError> errorCallback = new ExtensionErrorCallback<ExtensionError>() {
            @Override
            public void error(final ExtensionError extensionError) {
                // something went wrong, the listener couldn't be registered
            }
        };
        getApi().registerEventListener("com.adobe.eventType.configuration",
            "com.adobe.eventSource.requestContent", MyListener.class, errorCallback);
    }

    @Override
    public final String getName() {
        return "my.test.company";
    }

    @Override
    public final String getVersion() {
        return "1.0.0";
    }

    @Override
    public final void onUnregistered() {
        // extension unregistered successfully - perform cleanup
    }

    void handleConfigurationEvent(final Event event) {
        getExecutor().execute(new Runnable() {
            @Override
            public void run() {
                Log.d(getName(), String.format("Started processing new event of type %s and source %s", event.getType(), event.getSource()));
                // do your processing here
            }
        });
    }

    private ExecutorService getExecutor() {
        synchronized (executorMutex) {
            if (executor == null) {
                executor = Executors.newSingleThreadExecutor();
            }

            return executor;
        }
    }
}

{% endtab %}

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

iOS

  1. In Xcode, create a new file from the Cocoa Touch Class template and save it in your project.

  2. Name your class MyExtensionListener, and it should be a subclass to the ACPExtensionListener class.

    The MyExtensionListener.m file contains your extension interface declaration and imports ACPExtensionListener.h. In the example below, the methods that are available for overriding are also displayed:

MyExtensionListener.h

#import "ACPExtensionListener.h"
#import "ACPExtensionEvent.h"

@interface MyExtensionListener : ACPExtensionListener
    - (void) hear:(ACPExtensionEvent *)event;
@end
  1. At a minimum, you must provide an implementation for the hear method.

MyExtensionListener.m

#import "MyExtensionListener.h"
#import "MyExtension.h"

@implementation MyExtensionListener 
    - (void) hear:(ACPExtensionEvent *)event {
        MyExtension* parentExtension = [self getParentExtension];
        if (parentExtension == nil) {
            NSLog(@"The parent extension was nil, skipping event");
            return;
        }

        [parentExtension handleEvent:event];
}

/**
 * Returns the parent extension that owns this listener
 */
- (MyExtension*) getParentExtension {
    MyExtension* parentExtension = nil;
    if ([[self extension] isKindOfClass:MyExtension.class]) {
        parentExtension = (MyExtension*) [self extension];
    }

    return parentExtension;
}

@end

{% endtab %} {% endtabs %}

What can you do in your event handler?

Your listener has a reference to the parent extension that registered it. You can use this to centralize logic into your extension class and call into it from your listener. This also means you have access to the extension services API (ACPExtensionApi on iOS and ExtensionApi on Android) provided to the extension. This allows you to manage your shared states or register additional listeners. You also have access to the core SDK (ACPCore (iOS) or MobileCore (Android)). This reference allows you to dispatch events and receive responses.

Registering your event listener

You can register your listener class by using the init method as in the example below. In the hear method, you can access the registered extensions instance by using the extension property. This step can be useful to maintain state in your extension, or when you need to access to the ACPExtensionApi interface that the extension can also access.

The following example calls the listeners hear method when a change to the Adobe Experience Platform SDKs configuration occurs.

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

Android

Event listeners in Android are registered by using the registerEventListener method of the ExtensionApi interface. You can access this interface by using the getApi method in Extension. The following example shows how to register the listener from your extension constructor.

import com.adobe.marketing.mobile.Extension;
import com.adobe.marketing.mobile.ExtensionApi;

public class MyExtension extends Extension {

    public MyExtension(final ExtensionApi moduleApi) {
        super(moduleApi);

        ExtensionErrorCallback<ExtensionError> errorCallback = new ExtensionErrorCallback<ExtensionError>() {
            @Override
            public void error(final ExtensionError extensionError) {
                // something went wrong, the listener couldn't be registered
            }
        };
        getApi().registerEventListener("com.adobe.eventType.hub",
                "com.adobe.eventSource.sharedState", MyListener.class, errorCallback);
    }

    @Override
    public final String getName() {
        return "my.company";
    }

    @Override
    public final String getVersion() {
        return "1.0.0";
    }

    @Override
    public final void onUnregistered() {
        // extension unregistered successfully - perform cleanup
    }
}

{% endtab %}

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

iOS

In iOS the event listeners are registered using the registerListener method of the ACPExtensionApi interface. You can access this interface by using the api property in ACPExtension.

MyExtension.h

#import "ACPExtension.h"
#import "ACPExtensionEvent.h"

@interface MyExtension : ACPExtension

- (void) handleEvent: (ACPExtensionEvent*) event;

@end

MyExtension.m

#import "MyExtension.h"
#import "MyExtensionListener.h"

@implementation MyExtension 

- (instancetype) init {
    if (self = [super init]) {
        NSError *error = nil;

        // register a listener for configuration events
        if ([self.api registerListener:[MyExtensionListener class]
                             eventType:@"com.adobe.eventType.hub"
                           eventSource:@"com.adobe.eventSource.sharedState"
                                 error:&error]) {
            NSLog(@"MyExtensionListener successfully registered for Event Hub Shared State events");
        } else if (error) {
            NSLog(@"An error occured while registering MyExtensionListener, error code: %ld", [error code]);
        }
    }

    return self;
}

- (nullable NSString *) name {
    return @"my.company";
}

- (NSString *) version {
    return @"1.0.0";
}

- (void) onUnregister {
    [super onUnregister];
    // extension unregistered successfully - perform cleanup
}

- (void) handleEvent: (ACPExtensionEvent*) event {
    // process event
}
@end

{% endtab %} {% endtabs %}

Registering a wildcard listener

To listen for all events that are received and broadcasted by the Event Hub, register a wildcard listener by using the registerWildcardListener API.

{% hint style="warning" %} The hear method of this listener can be called often, and you should not do intensive processing on the same thread. You should use your own thread executor for the processing that you do in this listener. In doing so, the Event Hub is not blocked, and the listener is not unregistered if it takes too long. {% endhint %}

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

Android

import com.adobe.marketing.mobile.*;

public class MyExtension extends Extension {

    public MyExtension(final ExtensionApi moduleApi) {
        super(moduleApi);

        ExtensionErrorCallback<ExtensionError> errorCallback = new ExtensionErrorCallback<ExtensionError>() {
            @Override
            public void error(final ExtensionError extensionError) {
                // something went wrong, the listener couldn't be registered
            }
        };
        getApi().registerWildcardListener(MyListener.class, errorCallback);
    }

    ...
}

{% endtab %}

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

iOS

#import "MyExtension.h"
#import "MyExtensionWildcardListener.h"

@implementation MyExtension

-(instancetype) init {
    if (self = [super init]) {        
        NSError* error = nil;
        if ([[self api] registerWildcardListener:[MyExtensionWildcardListener class] 
                                             error:&error]) {
            NSLog(@"MyExtensionWildcardListener successfully registered");
        } else if (error) {
            NSLog(@"An error occurred while registering MyExtensionWildcardListener, error code: %ld", [error code]);
        }
    }

    return self;
}

{% endtab %} {% endtabs %}