YMPromptKit attempts to simplify your app code by providing flexible and extensible tools for soft prompting. Wait, what makes a prompt soft?
- Hard Prompt - A hard prompt is the typical dialog presented by iOS to obtain a user's permission for accessing a protected resource (such as their location or contacts). This prompt is considered hard because after showing it, the outcome cannot be easily changed.
- Soft Prompt - A prompt is soft when its results are not locked in. A soft prompt may be displayed several times, affording the application a second (or third or fourth) chance to present it's value proposition to the user.
The most well known hard prompt in iOS is that of push notifications. The value of these notifications cannot be overstated. Apps live and die by their ability – or lack of ability – to pop up highly relevant, immediately actionable alerts on your mobile device's screen. Unfortunately, once a user declines to allow you the right to push notifications at them, you have little recourse (few users will go through the trouble of manually re-enabling the feature). This has been discussed at length.
The easiest way to install is by using CocoaPods. The name of the pod is YMPromptKit.
pod 'YMPromptKit', '~> 1.0.0'
If you're not using CocoaPods, you need to add all of the classes in the YMPromptKit/
directory.
YMPromptKit depends on SDCAlertView by default (see next section).
SDCAlertView adds both functional value & reliability to YMPromptKit. However, if you cannot use SDCAlertView, one of it's dependencies, or are having other issues with it, you may opt to use iOS native prompting with a simple pod reference:
pod 'YMPromptKit/NativeAlerts', '~> 1.0.0`
YMPromptKit supports iOS 7.0 and higher.
Some features have degraded performance under iOS 7.0 due to reliance on APIs that were introduced in iOS 8.0. Where applicable, these differences are noted in the API comments.
The demo app is provided in the Demo
folder. Check it out.
A number of iOS soft prompts are provided out-of-the-box for immediate use. These are:
- Push Notifications
- Location Services
- Calendar Events & Reminders
- Recording / Microphone Access
- Photo Library
- Address Book & Contacts
Additional proprietary prompts can also be added as needed, such as requesting a user to sign in, or invite a friend.
The very first thing you need to do is include the header file for YMPromptKit and allocate space for an instance of YMPromptManager
#import <YMPromptKit/YMPromptKit.h>
@interface AppDelegate : NSObject <UIApplicationDelegate>
@property (nonatomic, readonly) YMPromptManager *promptManager;
@end
Now get a reference to a YMPromptManager
. The easiest way to do this is to create a prompt manager with all pre-configured prompt handlers already registered
_promptManager = [YMPromptManager createPromptManager];
The prompt manager requires a data source to tell it what to show the user. In this case, let's use ourself as the data source.
self.promptManager.dataSource = self;
To implement the data source, we need to provide one method to vend instances of YMPrompt
. For this example, let's pretend that we only care about push notifications. Add the following method:
- (YMPrompt *)promptManager:(YMPromptManager *)manager
promptForAccessType:(YMPromptAccessType)access
modes:(YMPromptAccessMode)modes {
if (access == kYMPromptAccessTypeNotifications) {
YMPrompt *prompt = [YMPrompt promptWithTitle:@"Stock Alerts"
message:@"Allow Yahoo Finance to send you important alerts"
@"about stocks you follow?"
grantTitle:@"Yes"
denyTitle:nil
contentView:nil];
return prompt;
}
return nil;
}
Now, when some code in your app asks the YMPromptManager
to request access for push notifications, the prompt manager will get an instance of YMPrompt
from us. The prompt we'll vend will have a custom title, message and grant permissions button label. The deny button title will not be specified; it will default to "Not Now".
Trigger the prompt from anywhere in your app, like this
[delegate.promptManager requestAccess:kYMPromptAccessTypeNotifications
modes:YMPromptNotificationOptionTypeAlert
completionHandler:^(BOOL softPromptAccepted, BOOL osPromptAccepted, YMPromptAuthorizationStatus status) {
if (softPromptAccepted) {
if (osPromptAccepted) {
// User accepted the soft prompt and the OS prompt. Win!
} else {
// User accepted the soft prompt, but denied the OS prompt. Lose :(
}
} else {
if (status == YMPromptAuthorizationStatusNotDetermined) {
// The user declined your soft prompt. Not a win, not a loss.
} else if (status == YMPromptAuthorizationStatusDenied) {
// Already denied access by the OS
} else if (status == YMPromptAuthorizationStatusRestricted) {
// Already authorized, but there is some OS restriction
} else if (status == YMPromptAuthorizationStatusAuthorized) {
// Already authorized by the OS
}
}
}];
For iOS 8, you must manually record the registration result with the prompt manager. In your app delegate record the notification registration result in -application: didRegisterUserNotificationSettings:
. This is also the point when you can register for remote notifications.
- (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings {
BOOL success = (notificationSettings.types != UIUserNotificationTypeNone);
[delegate.promptManager recordNotificationRegistrationResult:success];
// Permission was granted, register for remote notifications
[application registerForRemoteNotifications];
}
Soft prompts are pretty great, but one sure method to maximize user opt-out rates is to pester the user with requests too often. YMPromptKit can help you achieve an appropriate prompting cadence right out of the box. Let's ensure that our example app does not ask for push notification permission more than once in a 48 hour period, or more than 3 times.
Add a delegate method to give the prompt manager some application-specific guidance
- (BOOL)promptManager:(YMPromptManager *)manager shouldRequestAccessType:(YMPromptAccessType)accessType
modes:(YMPromptAccessMode)modes {
NSArray *promptLog = [manager.log promptHistory:accessType];
if (promptLog.count > 3) {
return NO; // enforce the request limit
} else if (promptLog.count) {
NSDictionary *dict = promptLog[0]; // data for the most recent history entry
NSDate *lastPromptDate = dict[kYMPromptLogDateKey];
NSTimeInterval interval = -3600 * 48; // 2 days, in seconds
if ([lastPromptDate timeIntervalSinceNow] > interval) {
return NO; // enforce the quiet period
}
}
return YES; // OK to display prompt!
}
The code above uses the prompt log to access the prompt history for the given access type. Each YMPromptManager
has a YMPromptLog
accessible via it's log
property. Default prompt managers – those returned by [YMPromptManager createPromptManager]
– always use the main prompt log available at [YMPromptLog mainLog]
.
Just before a prompt is displayed, a YMPromptManager
will add a date entry to it's YMPromptLog
instance. The prompt log, therefore, represents the recent history of soft prompts (last 15 occurances of each type of prompt, by default). The history is periodically flushed to disk and persists across sessions.
Although the dates of soft prompt occurances are enough to provide coarse control of cadence, it is each to imagine business requirements which cannot be satified by such limited data. To alleviate this issue, YMPromptLog
can accept arbitrary NSPropertyListSerialization-compatible metadata to store with each event entry.
Let's add to the example a requirement that the five sessions must elapse between push notification soft prompts. To do this, implement the YMPromptManagerDelegate
method:
- (NSDictionary *)promptManager:(YMPromptManager *)manager
willRequestAccessType:(YMPromptAccessType)accessType
modes:(YMPromptAccessMode)modes {
NSNumber *sessionNum = self.sessionNumber;
return @{ @"session_number": sessionNum };
}
Now, each event will have an associated dictionary that contains the session number of that it was displayed in. We can access this information and use it to apply application-specific logic from within the -promptManager:shouldRequestAccessType:modes:
delegate method. Extending the previous example of that delegate method from above:
- (BOOL)promptManager:(YMPromptManager *)manager shouldRequestAccessType:(YMPromptAccessType)accessType
modes:(YMPromptAccessMode)modes {
NSArray *promptLog = [manager.log promptHistory:accessType];
if (promptLog.count > 3) {
return NO; // enforce the request limit
} else if (promptLog.count) {
NSDictionary *dict = promptLog[0]; // data for the most recent history entry
NSDate *lastPromptDate = dict[kYMPromptLogDateKey];
NSTimeInterval interval = -3600 * 48; // 2 days, in seconds
if ([lastPromptDate timeIntervalSinceNow] > interval) {
return NO; // enforce the quiet period
} else {
#pragma >>> New example code starts here >>>
NSDictionary *appData = dict[kYMPromptLogUserInfoKey];
NSNumber *lastSessionNum = appData[@"session_number"];
NSNumber *sessionNum = self.sessionNumber;
if ([sessionNum integerValue] < [lastSessionNum integerValue] + 5) {
return NO; // enforce minimum sessions betwen prompting
}
}
}
return YES; // OK to display prompt!
}
YMPromptKit leverages SDCAlertView to provide flexible, native iOS-like alerts. One key benefit afforded by this design is that soft prompts may include an embedded custom view – even on iOS 7.
Let's add an image like the one shown above, using the running example:
- (YMPrompt *)promptManager:(YMPromptManager *)manager
promptForAccessType:(YMPromptAccessType)access
modes:(YMPromptAccessMode)modes {
UIImage *image = [UIImage imageNamed:@"push-notification.png"];
UIImageView *imageView = [[UIImageView alloc] initWithImage:image];
imageView.frame = CGRectMake(0, 0, 150, 150);
imageView.contentMode = UIViewContentModeScaleAspectFit;
if (access == kYMPromptAccessTypeNotifications) {
YMPrompt *prompt = [YMPrompt promptWithTitle:@"Stock Alerts"
message:@"Allow Yahoo Finance to send you important alerts"
@"about stocks you follow?"
grantTitle:@"Yes"
denyTitle:nil
contentView:imageView];
}
return prompt;
SDCAlertView – which powers YMPromptKit – works by presenting alerts as modal view controllers. When the alerts are dismissed (i.e. via it's buttons), SDCAlertView calls YMPromptKit completion blocks, which then call the local app's completion blocks. Beware: If you manually call -dismissModalViewController
, the completion blocks will not be triggered.
- Custom alert subviews are not supported. If specified, they will be silently ignored.
- iOS7 does not support block-based alerting, which requires internal state to be maintained while a soft prompt is being displayed. If clients attempt to present additional soft prompts prior to receiving a completion callback, this internal state may become corrupted, resulting in undefined behavior.
Report any bugs or send feature requests to the GitHub issues. Pull requests are very much welcomed. See CONTRIBUTING for details.
Apache 2.0 license. See the LICENSE file for details.