Implementing a Listener / Delegate for Custom User Attributes
This page provide details on how to set up a Listener / Delegate to fetch Custom User Attributes inside the application and process it
Context
The feature described in this page is particularly useful to fetch the user answers to a survey. The Screen Composer can be leveraged to publish user surveys (ie: onboarding questions, cancellation surveys etc...) that can be associated with a Custom User Attribute.
By implementing this feature, you will be able to:
- publish surveys in no-code using the Screen Composer
- automatically fetch the user data {attribute, type, value(s)}in the app
- and process it to send it to your backend or any 3 party integration
... without needing to update your app every time you publish a new survey.
To handle changes in user attributes, you can implement a listener (Android) or a delegate (iOS). This allows you to capture when a user attribute is set or removed within your app.
This feature is available starting from the following versions:
- iOS: v5.0.2+
- Android: v5.0.4+
- React Native: v5.0.4+
Functionning
When the user submit their answer(s) to the Survey, if a user attribute has been associated to the survey, this listener / delegate will automatically be called by the SDK to handover to the app.
After the app has been notified, it can:
- fetch the data {attribute ID, type, value(s)}
- and process it
Implementation
class UserAttributeHandler: PLYUserAttributeDelegate {
  func onUserAttributeSet(key: String, type: PLYUserAttributeType, value: Any?, source: PLYUserAttributeSource) {
    print("onUserAttributeSet: \(key) \(type) \(String(describing: value)) \(source)")
  }
  
  func onUserAttributeRemoved(key: String, source: PLYUserAttributeSource) {
    print("onUserAttributeRemoved: \(key) \(source)")
  }
}
Purchasely.setUserAttributeDelegate(UserAttributeHandler())
Purchasely.userAttributeListener = object : UserAttributeListener {
  override fun onUserAttributeSet(key: String, type: PLYUserAttributeType, value: Any, source: PLYUserAttributeSource) {
    Log.d(TAG, "User attribute added: $key, $type, $value from $source")
  }
  override fun onUserAttributeRemoved(key: String, source: PLYUserAttributeSource) {
    Log.d(TAG, "User attribute removed: $key")
  }
}
Purchasely.addUserAttributeSetListener((attribute: PurchaselyUserAttribute) => {
  console.log('Attribute set:', attribute);
});
Purchasely.addUserAttributeRemovedListener(attribute => {
  console.log('Attribute removed:', attribute);
});
// -- Definition of a Purchasely User Attribute -- //
export type PurchaselyUserAttribute = {
  key: string;
  value?: any | null;
  type?: PLYUserAttributeType | null;
  source?: PLYUserAttributeSource | null;
};
class MyUserAttributeListener implements UserAttributeListener {
  @override
  void onUserAttributeSet(String key, PLYUserAttributeType type, dynamic value, PLYUserAttributeSource source){
    print("Attribute set: $key, Type: $type, Value: $value, Source: $source");
  }
  @override
  void onUserAttributeRemoved(String key, PLYUserAttributeSource source) {
    print("Attribute removed: $key, Source: $source");
  }
}
Purchasely.setUserAttributeListener(MyUserAttributeListener());
The listener should be implemented for the following types of Custom User Attributes:
- String
- Int
- Float
- Bool
- Date
- Array of Strings
Surveys allowing multiple choice answers are associated with the type Array of Strings
When configuring your survey, you can define whether multiple answers are allowed or not.
- If the Survey is configured to allow ONE single answer, the attribute returned will be a String
- If the Survey is configured to allow MULTIPLE answers, the attribute returned will be an Array of Strings.
Once fetched, the data can be processed, which consists in sending it to your backend or any 3rd party integration directly from the app.
Understanding the PLYUserAttributeSource
PLYUserAttributeSourceThe source parameter of the listener/delegate methods indicates where the user attribute update originated. It can have two possible values:
- PURCHASELY: The change was initiated internally by the Purchasely’s SDK.
- CLIENT: The change was triggered directly by your app.
@objc public enum PLYUserAttributeSource: Int {
     case purchasely, client
 }
enum class PLYUserAttributeSource {
    PURCHASELY, 
    CLIENT
}
export enum PLYUserAttributeSource {
  PURCHASELY,
  CLIENT
}
enum PLYUserAttributeSource {
  purchasely,
  client,
}
This distinction helps you understand whether the attribute change was driven automatically by Purchasely's SDK or explicitly by your app's logic.
Ignore the delegate when the parameter
sourceis set toclientWhen your app sets a Custom User Attribute, the listener / delegate will be called back by the SDK with the parameter
sourceset toclient.To avoid processing a data that you already have - because the app has set it in the first place - in most cases, you should be able to ignore attribute triggered when the parameter
sourceis set toclient.
Updated 5 months ago