forked from organicmaps/organicmaps-tmp
[iOS] refactor local notifications
This commit is contained in:
parent
1961a8d507
commit
7bc097a9a0
14 changed files with 367 additions and 399 deletions
|
@ -23,6 +23,7 @@
|
|||
#import "BookmarksVC.h"
|
||||
#import "FacebookNativeAdAdapter.h"
|
||||
#import "LocalNotificationManager.h"
|
||||
#import "CoreNotificationWrapper.h"
|
||||
#import "MapViewController.h"
|
||||
#import "MWMActivityViewController.h"
|
||||
#import "MWMAlertViewController.h"
|
||||
|
|
11
iphone/Maps/Categories/NSDate+TimeDistance.h
Normal file
11
iphone/Maps/Categories/NSDate+TimeDistance.h
Normal file
|
@ -0,0 +1,11 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface NSDate (TimeDistance)
|
||||
|
||||
- (NSInteger)daysToNow;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
14
iphone/Maps/Categories/NSDate+TimeDistance.m
Normal file
14
iphone/Maps/Categories/NSDate+TimeDistance.m
Normal file
|
@ -0,0 +1,14 @@
|
|||
#import "NSDate+TimeDistance.h"
|
||||
|
||||
@implementation NSDate (TimeDistance)
|
||||
|
||||
- (NSInteger)daysToNow {
|
||||
NSDateComponents *components = [[NSCalendar currentCalendar] components:NSCalendarUnitDay
|
||||
fromDate:self
|
||||
toDate:[NSDate date]
|
||||
options:NSCalendarWrapComponents];
|
||||
|
||||
return components.day;
|
||||
}
|
||||
|
||||
@end
|
|
@ -1,5 +1,6 @@
|
|||
#import "MapsAppDelegate.h"
|
||||
|
||||
#import "CoreNotificationWrapper+Core.h"
|
||||
#import "EAGLView.h"
|
||||
#import "LocalNotificationManager.h"
|
||||
#import "MWMAuthorizationCommon.h"
|
||||
|
@ -13,6 +14,7 @@
|
|||
#import "MWMRouter.h"
|
||||
#import "MWMSearch+CoreSpotlight.h"
|
||||
#import "MWMTextToSpeech.h"
|
||||
#import "NSDate+TimeDistance.h"
|
||||
#import "MapViewController.h"
|
||||
#import "Statistics.h"
|
||||
#import "SwiftBridge.h"
|
||||
|
@ -131,11 +133,12 @@ void TrackMarketingAppLaunch()
|
|||
|
||||
using namespace osm_auth_ios;
|
||||
|
||||
@interface MapsAppDelegate ()<MWMFrameworkStorageObserver, UNUserNotificationCenterDelegate>
|
||||
@interface MapsAppDelegate ()<MWMFrameworkStorageObserver, NotificationManagerDelegate>
|
||||
|
||||
@property(nonatomic) NSInteger standbyCounter;
|
||||
@property(nonatomic) MWMBackgroundFetchScheduler * backgroundFetchScheduler;
|
||||
@property(nonatomic) id<IPendingTransactionsHandler> pendingTransactionHandler;
|
||||
@property(nonatomic) NotificationManager * notificationManager;
|
||||
|
||||
@end
|
||||
|
||||
|
@ -372,13 +375,6 @@ using namespace osm_auth_ios;
|
|||
|
||||
[self commonInit];
|
||||
|
||||
LocalNotificationManager * notificationManager = [LocalNotificationManager sharedManager];
|
||||
if (launchOptions[UIApplicationLaunchOptionsLocalNotificationKey])
|
||||
{
|
||||
NSNotification * notification = launchOptions[UIApplicationLaunchOptionsLocalNotificationKey];
|
||||
[notificationManager processNotification:notification.userInfo onLaunch:YES];
|
||||
}
|
||||
|
||||
if ([Alohalytics isFirstSession])
|
||||
{
|
||||
[self firstLaunchSetup];
|
||||
|
@ -401,7 +397,9 @@ using namespace osm_auth_ios;
|
|||
[GIDSignIn sharedInstance].clientID =
|
||||
[[NSBundle mainBundle] loadWithPlist:@"GoogleService-Info"][@"CLIENT_ID"];
|
||||
|
||||
[UNUserNotificationCenter currentNotificationCenter].delegate = self;
|
||||
self.notificationManager = [[NotificationManager alloc] init];
|
||||
self.notificationManager.delegate = self;
|
||||
[UNUserNotificationCenter currentNotificationCenter].delegate = self.notificationManager;
|
||||
|
||||
if ([MWMFrameworkHelper canUseNetwork]) {
|
||||
[[SubscriptionManager shared] validate];
|
||||
|
@ -429,57 +427,45 @@ using namespace osm_auth_ios;
|
|||
- (void)runBackgroundTasks:(NSArray<BackgroundFetchTask *> * _Nonnull)tasks
|
||||
completionHandler:(void (^_Nullable)(UIBackgroundFetchResult))completionHandler
|
||||
{
|
||||
auto completion = ^(UIBackgroundFetchResult result) {
|
||||
if (completionHandler)
|
||||
completionHandler(result);
|
||||
};
|
||||
self.backgroundFetchScheduler =
|
||||
[[MWMBackgroundFetchScheduler alloc] initWithTasks:tasks completionHandler:completion];
|
||||
[[MWMBackgroundFetchScheduler alloc] initWithTasks:tasks
|
||||
completionHandler:^(UIBackgroundFetchResult result) {
|
||||
if (completionHandler)
|
||||
completionHandler(result);
|
||||
}];
|
||||
[self.backgroundFetchScheduler run];
|
||||
}
|
||||
|
||||
- (void)application:(UIApplication *)application
|
||||
performFetchWithCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
|
||||
{
|
||||
auto onTap = ^{
|
||||
MapViewController * mapViewController = [MapViewController sharedController];
|
||||
[mapViewController.navigationController popToRootViewControllerAnimated:NO];
|
||||
[mapViewController showUGCAuth];
|
||||
};
|
||||
|
||||
if ([LocalNotificationManager.sharedManager showUGCNotificationIfNeeded:onTap])
|
||||
if ([LocalNotificationManager shouldShowAuthNotification])
|
||||
{
|
||||
AuthNotification * n = [[AuthNotification alloc] initWithTitle:L(@"notification_unsent_reviews_title")
|
||||
text:L(@"notification_unsent_reviews_message")];
|
||||
[self.notificationManager showNotification:n];
|
||||
[LocalNotificationManager authNotificationWasShown];
|
||||
completionHandler(UIBackgroundFetchResultNewData);
|
||||
return;
|
||||
}
|
||||
|
||||
lightweight::Framework const framework(lightweight::REQUEST_TYPE_NOTIFICATION);
|
||||
auto const notificationCandidate = framework.GetNotification();
|
||||
if (notificationCandidate)
|
||||
CoreNotificationWrapper * reviewNotification = [LocalNotificationManager reviewNotificationWrapper];
|
||||
if (reviewNotification)
|
||||
{
|
||||
auto const notification = notificationCandidate.get();
|
||||
if (notification.GetType() == notifications::NotificationCandidate::Type::UgcReview)
|
||||
{
|
||||
auto const place = notification.GetAddress().empty()
|
||||
? notification.GetReadableName()
|
||||
: notification.GetReadableName() + ", " + notification.GetAddress();
|
||||
|
||||
[LocalNotificationManager.sharedManager
|
||||
showReviewNotificationForPlace:@(place.c_str())
|
||||
onTap:^{
|
||||
[Statistics logEvent:kStatUGCReviewNotificationClicked];
|
||||
place_page::Info info;
|
||||
if (GetFramework().MakePlacePageInfo(notification, info))
|
||||
[[MapViewController sharedController].controlsManager showPlacePageReview:info];
|
||||
}];
|
||||
}
|
||||
NSString * text = reviewNotification.address.length > 0
|
||||
? [NSString stringWithFormat:@"%@, %@", reviewNotification.readableName, reviewNotification.address]
|
||||
: reviewNotification.readableName;
|
||||
ReviewNotification * n = [[ReviewNotification alloc] initWithTitle:L(@"notification_leave_review_v2_title")
|
||||
text:text
|
||||
notificationWrapper:reviewNotification];
|
||||
[self.notificationManager showNotification:n];
|
||||
[LocalNotificationManager reviewNotificationWasShown];
|
||||
}
|
||||
|
||||
auto tasks = @[
|
||||
[[MWMBackgroundStatisticsUpload alloc] init], [[MWMBackgroundEditsUpload alloc] init],
|
||||
[[MWMBackgroundUGCUpload alloc] init], [[MWMBackgroundDownloadMapNotification alloc] init]
|
||||
];
|
||||
[self runBackgroundTasks:tasks completionHandler:completionHandler];
|
||||
[self runBackgroundTasks:@[[MWMBackgroundStatisticsUpload new],
|
||||
[MWMBackgroundEditsUpload new],
|
||||
[MWMBackgroundUGCUpload new]]
|
||||
completionHandler:completionHandler];
|
||||
}
|
||||
|
||||
- (void)applicationDidReceiveMemoryWarning:(UIApplication *)application
|
||||
|
@ -752,45 +738,6 @@ continueUserActivity:(NSUserActivity *)userActivity
|
|||
};
|
||||
}
|
||||
|
||||
- (void)userNotificationCenter:(UNUserNotificationCenter *)center
|
||||
willPresentNotification:(UNNotification *)notification
|
||||
withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler
|
||||
{
|
||||
if ([LocalNotificationManager isLocalNotification:notification])
|
||||
completionHandler(UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionSound);
|
||||
else
|
||||
[MWMPushNotifications userNotificationCenter:center
|
||||
willPresentNotification:notification
|
||||
withCompletionHandler:completionHandler];
|
||||
}
|
||||
|
||||
- (void)userNotificationCenter:(UNUserNotificationCenter *)center
|
||||
didReceiveNotificationResponse:(UNNotificationResponse *)response
|
||||
withCompletionHandler:(void(^)(void))completionHandler
|
||||
{
|
||||
if ([LocalNotificationManager isLocalNotification:response.notification])
|
||||
{
|
||||
if ([response.actionIdentifier isEqualToString:UNNotificationDefaultActionIdentifier])
|
||||
{
|
||||
auto userInfo = response.notification.request.content.userInfo;
|
||||
[[LocalNotificationManager sharedManager] processNotification:userInfo onLaunch:NO];
|
||||
}
|
||||
completionHandler();
|
||||
}
|
||||
else
|
||||
{
|
||||
[MWMPushNotifications userNotificationCenter:center
|
||||
didReceiveNotificationResponse:response
|
||||
withCompletionHandler:completionHandler];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)application:(UIApplication *)application
|
||||
didReceiveLocalNotification:(UILocalNotification *)notification
|
||||
{
|
||||
[[LocalNotificationManager sharedManager] processNotification:notification.userInfo onLaunch:NO];
|
||||
}
|
||||
|
||||
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey, id> *)options
|
||||
{
|
||||
m_sourceApplication = options[UIApplicationOpenURLOptionsSourceApplicationKey];
|
||||
|
@ -805,7 +752,7 @@ didReceiveNotificationResponse:(UNNotificationResponse *)response
|
|||
if (isFBURL)
|
||||
return YES;
|
||||
|
||||
for (auto host in @[@"dlink.maps.me", @"dlink.mapsme.devmail.ru"])
|
||||
for (NSString * host in @[@"dlink.maps.me", @"dlink.mapsme.devmail.ru"])
|
||||
{
|
||||
if ([self checkLaunchURL:(url.host.length > 0 && [url.host rangeOfString:host].location != NSNotFound)
|
||||
? [self convertUniversalLink:url] : url])
|
||||
|
@ -926,8 +873,7 @@ didReceiveNotificationResponse:(UNNotificationResponse *)response
|
|||
return;
|
||||
|
||||
NSDate * lastLaunchDate = [standartDefaults objectForKey:kUDLastLaunchDateKey];
|
||||
NSUInteger daysFromLastLaunch = [self.class daysBetweenNowAndDate:lastLaunchDate];
|
||||
if (daysFromLastLaunch > 0)
|
||||
if (lastLaunchDate.daysToNow > 0)
|
||||
{
|
||||
sessionCount++;
|
||||
[standartDefaults setInteger:sessionCount forKey:kUDSessionsCountKey];
|
||||
|
@ -969,8 +915,7 @@ didReceiveNotificationResponse:(UNNotificationResponse *)response
|
|||
return NO;
|
||||
|
||||
NSDate * const lastRateRequestDate = [standartDefaults objectForKey:kUDLastRateRequestDate];
|
||||
NSUInteger const daysFromLastRateRequest =
|
||||
[MapsAppDelegate daysBetweenNowAndDate:lastRateRequestDate];
|
||||
NSInteger const daysFromLastRateRequest = lastRateRequestDate.daysToNow;
|
||||
// Do not show more than one alert per day.
|
||||
if (lastRateRequestDate != nil && daysFromLastRateRequest == 0)
|
||||
return NO;
|
||||
|
@ -1002,20 +947,6 @@ didReceiveNotificationResponse:(UNNotificationResponse *)response
|
|||
return YES;
|
||||
}
|
||||
|
||||
+ (NSInteger)daysBetweenNowAndDate:(NSDate *)fromDate
|
||||
{
|
||||
if (!fromDate)
|
||||
return 0;
|
||||
|
||||
NSDate * now = NSDate.date;
|
||||
NSCalendar * calendar = NSCalendar.currentCalendar;
|
||||
[calendar rangeOfUnit:NSCalendarUnitDay startDate:&fromDate interval:NULL forDate:fromDate];
|
||||
[calendar rangeOfUnit:NSCalendarUnitDay startDate:&now interval:NULL forDate:now];
|
||||
NSDateComponents * difference =
|
||||
[calendar components:NSCalendarUnitDay fromDate:fromDate toDate:now options:0];
|
||||
return difference.day;
|
||||
}
|
||||
|
||||
#pragma mark - Showcase
|
||||
|
||||
- (MWMMyTarget *)myTarget
|
||||
|
@ -1028,4 +959,25 @@ didReceiveNotificationResponse:(UNNotificationResponse *)response
|
|||
return _myTarget;
|
||||
}
|
||||
|
||||
#pragma mark - NotificationManagerDelegate
|
||||
|
||||
- (void)didOpenNotification:(Notification *)notification
|
||||
{
|
||||
if (notification.class == ReviewNotification.class)
|
||||
{
|
||||
[Statistics logEvent:kStatUGCReviewNotificationClicked];
|
||||
ReviewNotification * reviewNotification = (ReviewNotification *)notification;
|
||||
place_page::Info info;
|
||||
if (GetFramework().MakePlacePageInfo(reviewNotification.notificationWrapper.notificationCandidate, info))
|
||||
[[MapViewController sharedController].controlsManager showPlacePageReview:info];
|
||||
}
|
||||
else if (notification.class == AuthNotification.class)
|
||||
{
|
||||
[Statistics logEvent:@"UGC_UnsentNotification_clicked"];
|
||||
MapViewController * mapViewController = [MapViewController sharedController];
|
||||
[mapViewController.navigationController popToRootViewControllerAnimated:NO];
|
||||
[mapViewController showUGCAuth];
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
|
|
@ -60,17 +60,3 @@ final class BackgroundUGCUpload: BackgroundFetchTask {
|
|||
return "UGC upload"
|
||||
}
|
||||
}
|
||||
|
||||
@objc(MWMBackgroundDownloadMapNotification)
|
||||
final class BackgroundDownloadMapNotification: BackgroundFetchTask {
|
||||
override var frameworkType: BackgroundFetchTaskFrameworkType { return .full }
|
||||
|
||||
override fileprivate func fire() {
|
||||
LocalNotificationManager.shared().showDownloadMapNotificationIfNeeded(self.finish)
|
||||
}
|
||||
|
||||
override var description: String {
|
||||
return "Download map notification"
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
#include "Framework.h"
|
||||
|
||||
#include "base/sunrise_sunset.hpp"
|
||||
#include "platform/network_policy.hpp"
|
||||
#include "platform/network_policy_ios.h"
|
||||
|
||||
@implementation MWMFrameworkHelper
|
||||
|
||||
|
@ -99,8 +99,7 @@
|
|||
|
||||
+ (BOOL)canUseNetwork
|
||||
{
|
||||
return platform::GetCurrentNetworkPolicy().CanUse() ||
|
||||
GetPlatform().ConnectionStatus() != Platform::EConnectionType::CONNECTION_WWAN;
|
||||
return network_policy::CanUseNetwork();
|
||||
}
|
||||
|
||||
+ (BOOL)isNetworkConnected
|
||||
|
|
|
@ -0,0 +1,10 @@
|
|||
#import "CoreNotificationWrapper.h"
|
||||
|
||||
#include "map/notifications/notification_manager.hpp"
|
||||
|
||||
@interface CoreNotificationWrapper (Core)
|
||||
|
||||
- (instancetype)initWithNotificationCandidate:(notifications::NotificationCandidate const &)notification;
|
||||
- (notifications::NotificationCandidate)notificationCandidate;
|
||||
|
||||
@end
|
19
iphone/Maps/Core/Notifications/CoreNotificationWrapper.h
Normal file
19
iphone/Maps/Core/Notifications/CoreNotificationWrapper.h
Normal file
|
@ -0,0 +1,19 @@
|
|||
#import <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface CoreNotificationWrapper : NSObject
|
||||
|
||||
@property(nonatomic, copy) NSString *bestType;
|
||||
@property(nonatomic, copy) NSString *defaultName;
|
||||
@property(nonatomic, copy) NSString *readableName;
|
||||
@property(nonatomic, copy) NSString *address;
|
||||
@property(nonatomic) double x;
|
||||
@property(nonatomic) double y;
|
||||
|
||||
- (instancetype _Nullable)initWithNotificationDictionary:(NSDictionary *)dictionary;
|
||||
- (NSDictionary *)notificationDictionary;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
71
iphone/Maps/Core/Notifications/CoreNotificationWrapper.mm
Normal file
71
iphone/Maps/Core/Notifications/CoreNotificationWrapper.mm
Normal file
|
@ -0,0 +1,71 @@
|
|||
#import "CoreNotificationWrapper.h"
|
||||
#import "CoreNotificationWrapper+Core.h"
|
||||
|
||||
static NSString *const kX = @"x";
|
||||
static NSString *const kY = @"y";
|
||||
static NSString *const kType = @"type";
|
||||
static NSString *const kName = @"name";
|
||||
static NSString *const kReadableName = @"readableName";
|
||||
|
||||
@implementation CoreNotificationWrapper
|
||||
|
||||
- (instancetype)initWithNotificationDictionary:(NSDictionary *)dictionary {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
NSNumber *x = dictionary[kX];
|
||||
NSNumber *y = dictionary[kY];
|
||||
NSString *type = dictionary[kType];
|
||||
NSString *name = dictionary[kName];
|
||||
NSString *readableName = dictionary[kReadableName];
|
||||
if (x && y && type && name && readableName) {
|
||||
_x = x.doubleValue;
|
||||
_y = y.doubleValue;
|
||||
_bestType = type;
|
||||
_defaultName = name;
|
||||
_readableName = readableName;
|
||||
} else {
|
||||
return nil;
|
||||
}
|
||||
}
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
- (NSDictionary *)notificationDictionary {
|
||||
return @{
|
||||
kX : @(self.x),
|
||||
kY : @(self.y),
|
||||
kType : self.bestType,
|
||||
kName : self.defaultName,
|
||||
kReadableName : self.readableName
|
||||
};
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation CoreNotificationWrapper (Core)
|
||||
|
||||
- (instancetype)initWithNotificationCandidate:(notifications::NotificationCandidate const &)notification {
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_x = notification.GetPos().x;
|
||||
_y = notification.GetPos().y;
|
||||
_defaultName = @(notification.GetDefaultName().c_str());
|
||||
_bestType = @(notification.GetBestFeatureType().c_str());
|
||||
_readableName = @(notification.GetReadableName().c_str());
|
||||
_address = @(notification.GetAddress().c_str());
|
||||
}
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
- (notifications::NotificationCandidate)notificationCandidate {
|
||||
notifications::NotificationCandidate nc;
|
||||
nc.SetDefaultName(self.defaultName.UTF8String);
|
||||
nc.SetBestFeatureType(self.bestType.UTF8String);
|
||||
nc.SetPos(m2::PointD(self.x, self.y));
|
||||
nc.SetReadableName(self.readableName.UTF8String);
|
||||
return nc;
|
||||
}
|
||||
|
||||
@end
|
|
@ -1,18 +1,14 @@
|
|||
#import "MWMTypes.h"
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
#import <UserNotifications/UserNotifications.h>
|
||||
|
||||
typedef void (^CompletionHandler)(UIBackgroundFetchResult);
|
||||
@class CoreNotificationWrapper;
|
||||
|
||||
@interface LocalNotificationManager : NSObject
|
||||
|
||||
+ (instancetype)sharedManager;
|
||||
|
||||
+ (BOOL)isLocalNotification:(UNNotification *)notification;
|
||||
|
||||
- (BOOL)showUGCNotificationIfNeeded:(MWMVoidBlock)onTap;
|
||||
- (void)showReviewNotificationForPlace:(NSString *)place onTap:(MWMVoidBlock)onReviewNotification;
|
||||
- (void)showDownloadMapNotificationIfNeeded:(CompletionHandler)completionHandler;
|
||||
- (void)processNotification:(NSDictionary *)userInfo onLaunch:(BOOL)onLaunch;
|
||||
+ (BOOL)shouldShowAuthNotification;
|
||||
+ (void)authNotificationWasShown;
|
||||
+ (CoreNotificationWrapper * _Nullable)reviewNotificationWrapper;
|
||||
+ (void)reviewNotificationWasShown;
|
||||
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
|
|
|
@ -1,82 +1,23 @@
|
|||
#import "LocalNotificationManager.h"
|
||||
#import "CLLocation+Mercator.h"
|
||||
#import "MWMStorage.h"
|
||||
#import "MapViewController.h"
|
||||
#import "CoreNotificationWrapper+Core.h"
|
||||
#import "NSDate+TimeDistance.h"
|
||||
#import "Statistics.h"
|
||||
#import "SwiftBridge.h"
|
||||
#import "3party/Alohalytics/src/alohalytics_objc.h"
|
||||
|
||||
#include "Framework.h"
|
||||
|
||||
#include "storage/country_info_getter.hpp"
|
||||
#include "storage/storage_helpers.hpp"
|
||||
|
||||
#include "map/framework_light.hpp"
|
||||
|
||||
#include "platform/network_policy_ios.h"
|
||||
|
||||
namespace
|
||||
{
|
||||
NSString * const kLocalNotificationNameKey = @"LocalNotificationName";
|
||||
NSString * const kUGCNotificationValue = @"UGC";
|
||||
NSString * const kReviewNotificationValue = @"ReviewNotification";
|
||||
NSString * const kDownloadNotificationValue = @"Download";
|
||||
NSString * const kDownloadMapActionKey = @"DownloadMapActionKey";
|
||||
NSString * const kDownloadMapActionName = @"DownloadMapActionName";
|
||||
NSString * const kDownloadMapCountryId = @"DownloadMapCountryId";
|
||||
|
||||
NSString * const kFlagsKey = @"DownloadMapNotificationFlags";
|
||||
|
||||
NSTimeInterval constexpr kRepeatedNotificationIntervalInSeconds =
|
||||
3 * 30 * 24 * 60 * 60; // three months
|
||||
NSString * const kLastUGCNotificationDate = @"LastUGCNotificationDate";
|
||||
} // namespace
|
||||
|
||||
using namespace storage;
|
||||
|
||||
@interface LocalNotificationManager ()<CLLocationManagerDelegate, UIAlertViewDelegate>
|
||||
|
||||
@property(nonatomic) CLLocationManager * locationManager;
|
||||
@property(copy, nonatomic) CompletionHandler downloadMapCompletionHandler;
|
||||
@property(weak, nonatomic) NSTimer * timer;
|
||||
@property(copy, nonatomic) MWMVoidBlock onTap;
|
||||
@property(copy, nonatomic) MWMVoidBlock onReviewNotification;
|
||||
|
||||
@end
|
||||
static NSString * const kLastUGCNotificationDate = @"LastUGCNotificationDate";
|
||||
|
||||
@implementation LocalNotificationManager
|
||||
|
||||
+ (instancetype)sharedManager
|
||||
{
|
||||
static LocalNotificationManager * manager = nil;
|
||||
static dispatch_once_t onceToken;
|
||||
dispatch_once(&onceToken, ^{
|
||||
manager = [[self alloc] init];
|
||||
});
|
||||
return manager;
|
||||
}
|
||||
|
||||
+ (BOOL)isLocalNotification:(UNNotification *)notification
|
||||
{
|
||||
auto userInfo = notification.request.content.userInfo;
|
||||
return userInfo[kLocalNotificationNameKey] != nil;
|
||||
}
|
||||
|
||||
+ (BOOL)shouldShowUGCNotification
|
||||
+ (BOOL)shouldShowAuthNotification
|
||||
{
|
||||
if (!network_policy::CanUseNetwork())
|
||||
return NO;
|
||||
|
||||
|
||||
auto ud = [NSUserDefaults standardUserDefaults];
|
||||
if (NSDate * date = [ud objectForKey:kLastUGCNotificationDate])
|
||||
if (NSDate * date = [[NSUserDefaults standardUserDefaults] objectForKey:kLastUGCNotificationDate])
|
||||
{
|
||||
auto calendar = [NSCalendar currentCalendar];
|
||||
auto components = [calendar components:NSCalendarUnitDay fromDate:date
|
||||
toDate:[NSDate date] options:NSCalendarWrapComponents];
|
||||
|
||||
auto constexpr minDaysSinceLast = 5u;
|
||||
if (components.day <= minDaysSinceLast)
|
||||
if (date.daysToNow <= 5)
|
||||
return NO;
|
||||
}
|
||||
|
||||
|
@ -88,7 +29,7 @@ using namespace storage;
|
|||
return YES;
|
||||
}
|
||||
|
||||
+ (void)UGCNotificationWasShown
|
||||
+ (void)authNotificationWasShown
|
||||
{
|
||||
auto ud = [NSUserDefaults standardUserDefaults];
|
||||
[ud setObject:[NSDate date] forKey:kLastUGCNotificationDate];
|
||||
|
@ -96,212 +37,26 @@ using namespace storage;
|
|||
[Statistics logEvent:@"UGC_UnsentNotification_shown"];
|
||||
}
|
||||
|
||||
- (void)dealloc { _locationManager.delegate = nil; }
|
||||
- (void)processNotification:(NSDictionary *)userInfo onLaunch:(BOOL)onLaunch
|
||||
+ (CoreNotificationWrapper *)reviewNotificationWrapper
|
||||
{
|
||||
if ([userInfo[kDownloadMapActionKey] isEqualToString:kDownloadMapActionName])
|
||||
lightweight::Framework const framework(lightweight::REQUEST_TYPE_NOTIFICATION);
|
||||
auto const notificationCandidate = framework.GetNotification();
|
||||
if (notificationCandidate)
|
||||
{
|
||||
[Statistics logEvent:@"'Download Map' Notification Clicked"];
|
||||
MapViewController * mapViewController = [MapViewController sharedController];
|
||||
[mapViewController.navigationController popToRootViewControllerAnimated:NO];
|
||||
|
||||
NSString * notificationCountryId = userInfo[kDownloadMapCountryId];
|
||||
CountryId const countryId = notificationCountryId.UTF8String;
|
||||
[Statistics logEvent:kStatDownloaderMapAction
|
||||
withParameters:@{
|
||||
kStatAction : kStatDownload,
|
||||
kStatIsAuto : kStatNo,
|
||||
kStatFrom : kStatMap,
|
||||
kStatScenario : kStatDownload
|
||||
}];
|
||||
[MWMStorage downloadNode:countryId
|
||||
onSuccess:^{
|
||||
GetFramework().ShowNode(countryId);
|
||||
}];
|
||||
}
|
||||
else if ([userInfo[kLocalNotificationNameKey] isEqualToString:kUGCNotificationValue])
|
||||
{
|
||||
if (self.onTap)
|
||||
self.onTap();
|
||||
|
||||
[Statistics logEvent:@"UGC_UnsentNotification_clicked"];
|
||||
}
|
||||
else if ([userInfo[kLocalNotificationNameKey] isEqualToString:kReviewNotificationValue])
|
||||
{
|
||||
if (self.onReviewNotification)
|
||||
self.onReviewNotification();
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Location Notifications
|
||||
|
||||
- (void)showReviewNotificationForPlace:(NSString *)place onTap:(MWMVoidBlock)onReviewNotification {
|
||||
[Statistics logEvent:kStatUGCReviewNotificationShown];
|
||||
self.onReviewNotification = onReviewNotification;
|
||||
|
||||
UILocalNotification * notification = [[UILocalNotification alloc] init];
|
||||
notification.alertTitle = L(@"notification_leave_review_v2_title");
|
||||
notification.alertBody = place;
|
||||
notification.alertAction = L(@"leave_a_review");
|
||||
notification.soundName = UILocalNotificationDefaultSoundName;
|
||||
notification.userInfo = @{kLocalNotificationNameKey : kReviewNotificationValue};
|
||||
|
||||
[UIApplication.sharedApplication presentLocalNotificationNow:notification];
|
||||
}
|
||||
|
||||
- (BOOL)showUGCNotificationIfNeeded:(MWMVoidBlock)onTap
|
||||
{
|
||||
auto application = UIApplication.sharedApplication;
|
||||
auto identifier = UIBackgroundTaskInvalid;
|
||||
auto handler = [&identifier] {
|
||||
[[UIApplication sharedApplication] endBackgroundTask:identifier];
|
||||
};
|
||||
|
||||
identifier = [application beginBackgroundTaskWithExpirationHandler:^{
|
||||
handler();
|
||||
}];
|
||||
|
||||
if (![LocalNotificationManager shouldShowUGCNotification]) {
|
||||
handler();
|
||||
return NO;
|
||||
}
|
||||
|
||||
self.onTap = onTap;
|
||||
UILocalNotification * notification = [[UILocalNotification alloc] init];
|
||||
notification.alertTitle = L(@"notification_unsent_reviews_title");
|
||||
notification.alertBody = L(@"notification_unsent_reviews_message");
|
||||
notification.alertAction = L(@"authorization_button_sign_in");
|
||||
notification.soundName = UILocalNotificationDefaultSoundName;
|
||||
notification.userInfo = @{kLocalNotificationNameKey : kUGCNotificationValue};
|
||||
|
||||
[application presentLocalNotificationNow:notification];
|
||||
[LocalNotificationManager UGCNotificationWasShown];
|
||||
handler();
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void)showDownloadMapNotificationIfNeeded:(CompletionHandler)completionHandler
|
||||
{
|
||||
NSTimeInterval const completionTimeIndent = 2.0;
|
||||
NSTimeInterval const backgroundTimeRemaining =
|
||||
UIApplication.sharedApplication.backgroundTimeRemaining - completionTimeIndent;
|
||||
if ([CLLocationManager locationServicesEnabled] && backgroundTimeRemaining > 0.0)
|
||||
{
|
||||
self.downloadMapCompletionHandler = completionHandler;
|
||||
self.timer = [NSTimer scheduledTimerWithTimeInterval:backgroundTimeRemaining
|
||||
target:self
|
||||
selector:@selector(timerSelector:)
|
||||
userInfo:nil
|
||||
repeats:NO];
|
||||
LOG(LINFO, ("startUpdatingLocation"));
|
||||
[self.locationManager startUpdatingLocation];
|
||||
}
|
||||
else
|
||||
{
|
||||
LOG(LINFO, ("stopUpdatingLocation"));
|
||||
[self.locationManager stopUpdatingLocation];
|
||||
completionHandler(UIBackgroundFetchResultFailed);
|
||||
}
|
||||
}
|
||||
|
||||
- (BOOL)shouldShowNotificationForCountryId:(NSString *)countryId
|
||||
{
|
||||
if (!countryId || countryId.length == 0)
|
||||
return NO;
|
||||
NSUserDefaults * ud = NSUserDefaults.standardUserDefaults;
|
||||
NSDictionary<NSString *, NSDate *> * flags = [ud objectForKey:kFlagsKey];
|
||||
NSDate * lastShowDate = flags[countryId];
|
||||
return !lastShowDate ||
|
||||
[[NSDate date] timeIntervalSinceDate:lastShowDate] >
|
||||
kRepeatedNotificationIntervalInSeconds;
|
||||
}
|
||||
|
||||
- (void)markNotificationShownForCountryId:(NSString *)countryId
|
||||
{
|
||||
NSUserDefaults * ud = NSUserDefaults.standardUserDefaults;
|
||||
NSMutableDictionary<NSString *, NSDate *> * flags = [[ud objectForKey:kFlagsKey] mutableCopy];
|
||||
if (!flags)
|
||||
flags = [NSMutableDictionary dictionary];
|
||||
flags[countryId] = [NSDate date];
|
||||
[ud setObject:flags forKey:kFlagsKey];
|
||||
[ud synchronize];
|
||||
}
|
||||
|
||||
- (void)timerSelector:(id)sender
|
||||
{
|
||||
// Location still was not received but it's time to finish up so system will not kill us.
|
||||
LOG(LINFO, ("stopUpdatingLocation"));
|
||||
[self.locationManager stopUpdatingLocation];
|
||||
[self performCompletionHandler:UIBackgroundFetchResultFailed];
|
||||
}
|
||||
|
||||
- (void)performCompletionHandler:(UIBackgroundFetchResult)result
|
||||
{
|
||||
if (!self.downloadMapCompletionHandler)
|
||||
return;
|
||||
self.downloadMapCompletionHandler(result);
|
||||
self.downloadMapCompletionHandler = nil;
|
||||
}
|
||||
|
||||
#pragma mark - Location Manager
|
||||
|
||||
- (CLLocationManager *)locationManager
|
||||
{
|
||||
if (!_locationManager)
|
||||
{
|
||||
_locationManager = [[CLLocationManager alloc] init];
|
||||
_locationManager.delegate = self;
|
||||
_locationManager.distanceFilter = kCLLocationAccuracyThreeKilometers;
|
||||
}
|
||||
return _locationManager;
|
||||
}
|
||||
|
||||
- (void)locationManager:(CLLocationManager *)manager
|
||||
didUpdateLocations:(NSArray<CLLocation *> *)locations
|
||||
{
|
||||
[self.timer invalidate];
|
||||
LOG(LINFO, ("stopUpdatingLocation"));
|
||||
[self.locationManager stopUpdatingLocation];
|
||||
NSString * flurryEventName = @"'Download Map' Notification Didn't Schedule";
|
||||
UIBackgroundFetchResult result = UIBackgroundFetchResultNoData;
|
||||
|
||||
BOOL const inBackground =
|
||||
UIApplication.sharedApplication.applicationState == UIApplicationStateBackground;
|
||||
BOOL const onWiFi = (Platform::ConnectionStatus() == Platform::EConnectionType::CONNECTION_WIFI);
|
||||
if (inBackground && onWiFi)
|
||||
{
|
||||
CLLocation * lastLocation = locations.lastObject;
|
||||
auto const & mercator = lastLocation.mercator;
|
||||
auto & f = GetFramework();
|
||||
auto const & countryInfoGetter = f.GetCountryInfoGetter();
|
||||
if (!IsPointCoveredByDownloadedMaps(mercator, f.GetStorage(), countryInfoGetter))
|
||||
auto const notification = notificationCandidate.get();
|
||||
if (notification.GetType() == notifications::NotificationCandidate::Type::UgcReview)
|
||||
{
|
||||
NSString * countryId = @(countryInfoGetter.GetRegionCountryId(mercator).c_str());
|
||||
if ([self shouldShowNotificationForCountryId:countryId])
|
||||
{
|
||||
[self markNotificationShownForCountryId:countryId];
|
||||
|
||||
UILocalNotification * notification = [[UILocalNotification alloc] init];
|
||||
notification.alertAction = L(@"download");
|
||||
notification.alertBody = L(@"download_map_notification");
|
||||
notification.soundName = UILocalNotificationDefaultSoundName;
|
||||
notification.userInfo =
|
||||
@{kLocalNotificationNameKey : kDownloadNotificationValue,
|
||||
kDownloadMapActionKey : kDownloadMapActionName,
|
||||
kDownloadMapCountryId : countryId};
|
||||
|
||||
UIApplication * application = UIApplication.sharedApplication;
|
||||
[application presentLocalNotificationNow:notification];
|
||||
|
||||
[Alohalytics logEvent:@"suggestedToDownloadMissingMapForCurrentLocation"
|
||||
atLocation:lastLocation];
|
||||
flurryEventName = @"'Download Map' Notification Scheduled";
|
||||
result = UIBackgroundFetchResultNewData;
|
||||
}
|
||||
CoreNotificationWrapper * w = [[CoreNotificationWrapper alloc] initWithNotificationCandidate:notification];
|
||||
return w;
|
||||
}
|
||||
}
|
||||
[Statistics logEvent:flurryEventName withParameters:@{ @"WiFi" : @(onWiFi) }];
|
||||
[self performCompletionHandler:result];
|
||||
|
||||
return nil;
|
||||
}
|
||||
|
||||
+ (void)reviewNotificationWasShown
|
||||
{
|
||||
[Statistics logEvent:kStatUGCReviewNotificationShown];
|
||||
}
|
||||
|
||||
@end
|
||||
|
|
49
iphone/Maps/Core/Notifications/NotificationManager.swift
Normal file
49
iphone/Maps/Core/Notifications/NotificationManager.swift
Normal file
|
@ -0,0 +1,49 @@
|
|||
@objc
|
||||
protocol NotificationManagerDelegate {
|
||||
func didOpenNotification(_ notification: Notification)
|
||||
}
|
||||
|
||||
@objc
|
||||
final class NotificationManager: NSObject {
|
||||
@objc var delegate: NotificationManagerDelegate?
|
||||
|
||||
@objc
|
||||
func showNotification(_ notification: Notification) {
|
||||
let notificationContent = UNMutableNotificationContent()
|
||||
notificationContent.title = notification.title
|
||||
notificationContent.body = notification.text
|
||||
notificationContent.sound = UNNotificationSound.default()
|
||||
notificationContent.userInfo = notification.userInfo
|
||||
let notificationRequest = UNNotificationRequest(identifier: notification.identifier,
|
||||
content: notificationContent,
|
||||
trigger: UNTimeIntervalNotificationTrigger(timeInterval: 1,
|
||||
repeats: false))
|
||||
UNUserNotificationCenter.current().add(notificationRequest)
|
||||
}
|
||||
}
|
||||
|
||||
extension NotificationManager: UNUserNotificationCenterDelegate {
|
||||
func userNotificationCenter(_ center: UNUserNotificationCenter,
|
||||
willPresent notification: UNNotification,
|
||||
withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
|
||||
if Notification.fromUserInfo(notification.request.content.userInfo) != nil {
|
||||
completionHandler([.sound, .alert])
|
||||
} else {
|
||||
MWMPushNotifications.userNotificationCenter(center,
|
||||
willPresent: notification,
|
||||
withCompletionHandler: completionHandler)
|
||||
}
|
||||
}
|
||||
|
||||
func userNotificationCenter(_ center: UNUserNotificationCenter,
|
||||
didReceive response: UNNotificationResponse,
|
||||
withCompletionHandler completionHandler: @escaping () -> Void) {
|
||||
if let n = Notification.fromUserInfo(response.notification.request.content.userInfo) {
|
||||
delegate?.didOpenNotification(n)
|
||||
} else {
|
||||
MWMPushNotifications.userNotificationCenter(center,
|
||||
didReceive: response,
|
||||
withCompletionHandler: completionHandler)
|
||||
}
|
||||
}
|
||||
}
|
83
iphone/Maps/Core/Notifications/Notifications.swift
Normal file
83
iphone/Maps/Core/Notifications/Notifications.swift
Normal file
|
@ -0,0 +1,83 @@
|
|||
@objc
|
||||
class Notification: NSObject {
|
||||
fileprivate static let kLocalNotificationName = "LocalNotificationName"
|
||||
fileprivate static let kLocalNotificationTitle = "LocalNotificationTitle"
|
||||
fileprivate static let kLocalNotificationText = "LocalNotificationText"
|
||||
let title: String
|
||||
let text: String
|
||||
var userInfo: [AnyHashable : Any] {
|
||||
return [Notification.kLocalNotificationName : identifier,
|
||||
Notification.kLocalNotificationTitle : title,
|
||||
Notification.kLocalNotificationText : text]
|
||||
}
|
||||
|
||||
var identifier: String {
|
||||
assert(false, "Override in subclass")
|
||||
return ""
|
||||
}
|
||||
|
||||
@objc
|
||||
init(title: String, text: String) {
|
||||
self.title = title
|
||||
self.text = text
|
||||
}
|
||||
|
||||
convenience init?(userInfo: [AnyHashable : Any]) {
|
||||
if let title = userInfo[Notification.kLocalNotificationTitle] as? String,
|
||||
let text = userInfo[Notification.kLocalNotificationText] as? String {
|
||||
self.init(title: title, text: text)
|
||||
} else {
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
class func fromUserInfo(_ userInfo: [AnyHashable : Any]) -> Notification? {
|
||||
if let notificationName = userInfo[Notification.kLocalNotificationName] as? String {
|
||||
switch notificationName {
|
||||
case ReviewNotification.identifier:
|
||||
return ReviewNotification(userInfo: userInfo)
|
||||
case AuthNotification.identifier:
|
||||
return AuthNotification(userInfo: userInfo)
|
||||
default:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
@objc
|
||||
class AuthNotification: Notification {
|
||||
static let identifier = "UGC"
|
||||
override var identifier: String { return AuthNotification.identifier }
|
||||
}
|
||||
|
||||
@objc
|
||||
class ReviewNotification: Notification {
|
||||
private static let kNotificationObject = "NotificationObject"
|
||||
static let identifier = "ReviewNotification"
|
||||
@objc let notificationWrapper: CoreNotificationWrapper
|
||||
override var identifier: String { return ReviewNotification.identifier }
|
||||
override var userInfo: [AnyHashable : Any] {
|
||||
var result = super.userInfo
|
||||
result[ReviewNotification.kNotificationObject] = notificationWrapper.notificationDictionary()
|
||||
return result
|
||||
}
|
||||
|
||||
@objc
|
||||
init(title: String, text: String, notificationWrapper: CoreNotificationWrapper) {
|
||||
self.notificationWrapper = notificationWrapper
|
||||
super.init(title: title, text: text)
|
||||
}
|
||||
|
||||
convenience init?(userInfo: [AnyHashable : Any]) {
|
||||
if let title = userInfo[Notification.kLocalNotificationTitle] as? String,
|
||||
let text = userInfo[Notification.kLocalNotificationText] as? String,
|
||||
let notificationDictionary = userInfo[ReviewNotification.kNotificationObject] as? [AnyHashable : Any],
|
||||
let notificationWrapper = CoreNotificationWrapper(notificationDictionary: notificationDictionary) {
|
||||
self.init(title: title, text: text, notificationWrapper: notificationWrapper)
|
||||
} else {
|
||||
return nil
|
||||
}
|
||||
}
|
||||
}
|
|
@ -395,6 +395,8 @@
|
|||
4767CDA620AB1F6200BD8166 /* LeftAlignedIconButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4767CDA520AB1F6200BD8166 /* LeftAlignedIconButton.swift */; };
|
||||
4767CDA820AB401000BD8166 /* LinkTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4767CDA720AB401000BD8166 /* LinkTextView.swift */; };
|
||||
4767CDC120B477BA00BD8166 /* WelcomeViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4767CDC020B477BA00BD8166 /* WelcomeViewController.swift */; };
|
||||
47699A0821F08E37009E6585 /* NSDate+TimeDistance.m in Sources */ = {isa = PBXBuildFile; fileRef = 47699A0721F08E37009E6585 /* NSDate+TimeDistance.m */; };
|
||||
47699A0A21F0C4C8009E6585 /* NotificationManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47699A0921F0C4C8009E6585 /* NotificationManager.swift */; };
|
||||
477D7AC7218F1515007EE2CB /* IPaidRoutePurchase.swift in Sources */ = {isa = PBXBuildFile; fileRef = 477D7AC6218F1515007EE2CB /* IPaidRoutePurchase.swift */; };
|
||||
47800D1D20BEEE2E00072F42 /* TermsOfUseController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47800D1C20BEEE2E00072F42 /* TermsOfUseController.swift */; };
|
||||
47800D2520C05E3200072F42 /* libFlurry_8.6.1.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 47800D2420C05E3200072F42 /* libFlurry_8.6.1.a */; };
|
||||
|
@ -429,6 +431,8 @@
|
|||
47E6CB0C2178BA3600EA102B /* SearchBannerCell.xib in Resources */ = {isa = PBXBuildFile; fileRef = 47E6CB0A2178BA3600EA102B /* SearchBannerCell.xib */; };
|
||||
47EF05B321504D8F00EAC269 /* RemoveAdsPresentationController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47EF05B221504D8F00EAC269 /* RemoveAdsPresentationController.swift */; };
|
||||
47F67D1521CAB21B0069754E /* MWMImageCoder.m in Sources */ = {isa = PBXBuildFile; fileRef = 47F67D1421CAB21B0069754E /* MWMImageCoder.m */; };
|
||||
47F6E51221F61908004580CA /* CoreNotificationWrapper.mm in Sources */ = {isa = PBXBuildFile; fileRef = 47F6E51121F61908004580CA /* CoreNotificationWrapper.mm */; };
|
||||
47F6E51721FB3C51004580CA /* Notifications.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47F6E51621FB3C51004580CA /* Notifications.swift */; };
|
||||
47F86CFF20C936FC00FEE291 /* TabView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47F86CFE20C936FC00FEE291 /* TabView.swift */; };
|
||||
47F86D0120C93D8D00FEE291 /* TabViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47F86D0020C93D8D00FEE291 /* TabViewController.swift */; };
|
||||
4A300ED51C6DCFD400140018 /* countries-strings in Resources */ = {isa = PBXBuildFile; fileRef = 4A300ED31C6DCFD400140018 /* countries-strings */; };
|
||||
|
@ -1426,6 +1430,9 @@
|
|||
4767CDA520AB1F6200BD8166 /* LeftAlignedIconButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LeftAlignedIconButton.swift; sourceTree = "<group>"; };
|
||||
4767CDA720AB401000BD8166 /* LinkTextView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkTextView.swift; sourceTree = "<group>"; };
|
||||
4767CDC020B477BA00BD8166 /* WelcomeViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WelcomeViewController.swift; sourceTree = "<group>"; };
|
||||
47699A0621F08E37009E6585 /* NSDate+TimeDistance.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "NSDate+TimeDistance.h"; sourceTree = "<group>"; };
|
||||
47699A0721F08E37009E6585 /* NSDate+TimeDistance.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "NSDate+TimeDistance.m"; sourceTree = "<group>"; };
|
||||
47699A0921F0C4C8009E6585 /* NotificationManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NotificationManager.swift; sourceTree = "<group>"; };
|
||||
477D7AC6218F1515007EE2CB /* IPaidRoutePurchase.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = IPaidRoutePurchase.swift; sourceTree = "<group>"; };
|
||||
47800D1C20BEEE2E00072F42 /* TermsOfUseController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TermsOfUseController.swift; sourceTree = "<group>"; };
|
||||
47800D2420C05E3200072F42 /* libFlurry_8.6.1.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; path = libFlurry_8.6.1.a; sourceTree = "<group>"; };
|
||||
|
@ -1473,6 +1480,10 @@
|
|||
47F67D1321CAB21B0069754E /* MWMImageCoder.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = MWMImageCoder.h; sourceTree = "<group>"; };
|
||||
47F67D1421CAB21B0069754E /* MWMImageCoder.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = MWMImageCoder.m; sourceTree = "<group>"; };
|
||||
47F67D1621CAB50B0069754E /* IMWMImageCoder.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = IMWMImageCoder.h; sourceTree = "<group>"; };
|
||||
47F6E51021F61908004580CA /* CoreNotificationWrapper.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = CoreNotificationWrapper.h; sourceTree = "<group>"; };
|
||||
47F6E51121F61908004580CA /* CoreNotificationWrapper.mm */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.objcpp; path = CoreNotificationWrapper.mm; sourceTree = "<group>"; };
|
||||
47F6E51321F61974004580CA /* CoreNotificationWrapper+Core.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "CoreNotificationWrapper+Core.h"; sourceTree = "<group>"; };
|
||||
47F6E51621FB3C51004580CA /* Notifications.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Notifications.swift; sourceTree = "<group>"; };
|
||||
47F86CFE20C936FC00FEE291 /* TabView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabView.swift; sourceTree = "<group>"; };
|
||||
47F86D0020C93D8D00FEE291 /* TabViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabViewController.swift; sourceTree = "<group>"; };
|
||||
4A00DBDE1AB704C400113624 /* drules_proto_dark.bin */ = {isa = PBXFileReference; lastKnownFileType = archive.macbinary; name = drules_proto_dark.bin; path = ../../data/drules_proto_dark.bin; sourceTree = "<group>"; };
|
||||
|
@ -2685,6 +2696,8 @@
|
|||
33E905452180C40900868CAC /* UIViewController+Authorization.swift */,
|
||||
347E03981FAC5F1D00426032 /* UIWindow+InputLanguage.swift */,
|
||||
4767CDA320AAF66B00BD8166 /* NSAttributedString+HTML.swift */,
|
||||
47699A0621F08E37009E6585 /* NSDate+TimeDistance.h */,
|
||||
47699A0721F08E37009E6585 /* NSDate+TimeDistance.m */,
|
||||
);
|
||||
path = Categories;
|
||||
sourceTree = "<group>";
|
||||
|
@ -2916,6 +2929,11 @@
|
|||
3486B5061E27A4B50069C126 /* LocalNotificationManager.mm */,
|
||||
3486B50A1E27A4C50069C126 /* MWMPushNotifications.h */,
|
||||
3486B50B1E27A6DA0069C126 /* MWMPushNotifications.mm */,
|
||||
47699A0921F0C4C8009E6585 /* NotificationManager.swift */,
|
||||
47F6E51021F61908004580CA /* CoreNotificationWrapper.h */,
|
||||
47F6E51121F61908004580CA /* CoreNotificationWrapper.mm */,
|
||||
47F6E51321F61974004580CA /* CoreNotificationWrapper+Core.h */,
|
||||
47F6E51621FB3C51004580CA /* Notifications.swift */,
|
||||
);
|
||||
path = Notifications;
|
||||
sourceTree = "<group>";
|
||||
|
@ -5057,6 +5075,7 @@
|
|||
342CC5F21C2D7730005F3FE5 /* MWMAuthorizationLoginViewController.mm in Sources */,
|
||||
340475591E081A4600C92850 /* WebViewController.mm in Sources */,
|
||||
3404F4992028A20D0090E401 /* BMCCategoryCell.swift in Sources */,
|
||||
47F6E51721FB3C51004580CA /* Notifications.swift in Sources */,
|
||||
F62607FD207B790300176C5A /* SpinnerAlert.swift in Sources */,
|
||||
34F407411E9E1AFF00E57AC0 /* RBBanner.swift in Sources */,
|
||||
3444DFD21F17620C00E73099 /* MWMMapWidgetsHelper.mm in Sources */,
|
||||
|
@ -5213,6 +5232,7 @@
|
|||
47F67D1521CAB21B0069754E /* MWMImageCoder.m in Sources */,
|
||||
34AB66861FC5AA330078E451 /* MWMNavigationInfoView.mm in Sources */,
|
||||
34C9BD051C6DB693000DC38D /* MWMViewController.mm in Sources */,
|
||||
47699A0A21F0C4C8009E6585 /* NotificationManager.swift in Sources */,
|
||||
331630D12191D74B00BB91A9 /* TagSectionHeaderView.swift in Sources */,
|
||||
F6E2FDA41E097BA00083EBEC /* MWMCuisineEditorViewController.mm in Sources */,
|
||||
3454D7CB1E07F045004AF2AD /* UIColor+MapsMeColor.mm in Sources */,
|
||||
|
@ -5234,6 +5254,7 @@
|
|||
3454D7BF1E07F045004AF2AD /* DateComponentsFormatter+ETA.swift in Sources */,
|
||||
6741AA1C1BF340DE002C974C /* MWMRoutingDisclaimerAlert.mm in Sources */,
|
||||
34D3B0481E389D05004100F9 /* MWMNoteCell.mm in Sources */,
|
||||
47F6E51221F61908004580CA /* CoreNotificationWrapper.mm in Sources */,
|
||||
F6E2FE971E097BA00083EBEC /* ContextViews.mm in Sources */,
|
||||
344532561F7155540059FBCC /* UGCReviewModel.swift in Sources */,
|
||||
6741AA1D1BF340DE002C974C /* MWMDownloadTransitMapAlert.mm in Sources */,
|
||||
|
@ -5298,6 +5319,7 @@
|
|||
F6A2184A1CA3F26800BE2CC6 /* MWMEditorViralActivityItem.mm in Sources */,
|
||||
F6E2FED61E097BA00083EBEC /* MWMSearchChangeModeView.mm in Sources */,
|
||||
33603C85219F0F6300B11FFE /* SharingPropertiesViewController.swift in Sources */,
|
||||
47699A0821F08E37009E6585 /* NSDate+TimeDistance.m in Sources */,
|
||||
34845DB31E165E24003D55B9 /* SearchNoResultsViewController.swift in Sources */,
|
||||
34AB660B1FC5AA320078E451 /* MWMNavigationDashboardEntity.mm in Sources */,
|
||||
F5BD29FF26AD58255766C51A /* DiscoverySpinnerCell.swift in Sources */,
|
||||
|
|
Loading…
Add table
Reference in a new issue