iphone - iOS 角标(Badge)编号实时更新

标签 iphone ios xcode ipad sdk

我为 iOS 创建了一个自定义日历,我正在尝试使用角标(Badge)编号来显示当天的编号,但一天过去后编号没有改变,我的意思是他们应该及时更新,这是我的代码:我需要诸如天气实时应用程序之类的东西,此应用程序不发送任何推送通知!

    int a = [DateComponents showDay];
    [UIApplication sharedApplication].applicationIconBadgeNumber = a ;

最佳答案

目前在 iOS 中做到这一点并不容易。您可以接近,但不能每天可靠地更新角标(Badge),除非用户偶尔打开应用程序。

您需要使用 UILocalNotification 。您可以创建和安排一个“静默”通知来更新应用程序角标(Badge),而不用提醒用户或播放这样的警报声音:

UILocalNotification* notification = [[UILocalNotification alloc] init];
notification.fireDate = [NSDate dateWithTimeIntervalSinceNow:seconds];
notification.timeZone = [NSTimeZone systemTimeZone];
notification.alertBody = nil;
notification.soundName = nil;
notification.applicationIconBadgeNumber = dayTomorrow;

[[UIApplication sharedApplication] scheduleLocalNotification:notification];
[notification release];

其中 dayTomorrow 是一个月中明天的整数。 seconds 是某个时间间隔 - 这可能类似于午夜的间隔。传递此通知时,用户将不会收到警报或声音,但应用程序角标(Badge)应更新为新值。您可以使用 repeatInterval 属性每天重复此通知,但这里的问题是第二天您需要将 dayTomorrow 更改为不同的值,并在之后的第二天再次更改。但是重复的警报在 applicationIconBadgeNumber 中将始终具有相同的值。

所以我认为实现任何接近你想要的东西的唯一方法是安排多个本地通知——你最多可以提前安排 64 个——每一个间隔一天,并且每个都将当天的值设置为 applicationIconBadgeNumber 。这些必须是非重复的,因此请确保将 repeatInterval 设置为 nil(或不要设置它,因为 nil 是默认值)。假设所有这些通知都由 iOS 可靠地传送,您将能够在长达 64 天的时间内静默更新角标(Badge)编号,而不会打扰用户。在那之后,角标(Badge)将停止更新... 除非 您设法在此期间安排更多通知 - 即当用户打开应用程序时。您可以尝试做的是在应用启动时取消所有现有的预定通知:
[[UIApplication sharedApplication] cancelAllLocalNotifications];

然后提前 64 天循环,将通知安排在每天午夜,并使用 applicationIconBadgeNumber 属性的正确日期。只要用户至少每 64 天打开一次您的应用程序,您就会使角标(Badge)图标保持最新状态。如果您设想您的应用程序用户频繁打开应用程序(例如每天多次),那么优化可能是在取消之前检查现有通知的数量并创建 64 个新通知,如下所示:
if ([[[UIApplication sharedApplication] scheduledLocalNotifications] count] < 10)
{
    //there are 10 or fewer days' worth of notifications scheduled, so create and
    //schedule more here, up to 64 in total.
}

其他几点需要注意:
  • 我不确定本地通知的可靠性。不能保证推送通知,但我希望并期望本地通知能保证送达。例如如果手机在午夜关闭并且只在早上 7 点打开,我希望安排在午夜的本地通知将立即发送。但我从来没有测试过这个。
  • 如果您的应用程序在发送预定通知时当前处于打开状态,则会调用 application:didReceiveLocalNotification:,但不会更新角标(Badge)图标。因此,为了适应这种情况,我建议每次启动应用程序时(或从后台返回到前台)将角标(Badge)图标更新为当天。
  • 不确定 Apple 会如何使用角标(Badge)图标。在审查您的应用程序时,他们可能会觉得这让用户感到困惑,因为它通常用于指示应用程序中新/更改数据的项目数。因此,您的应用可能会因此被拒绝。
  • 理想情况下,Apple 将为开发人员提供一个 API,以便向用户提供这种“一目了然”的信息,无论是通过以某种方式更改应用程序图标/角标(Badge),还是通过小部件等来实现的。我上面描述的应该可以工作,但显然有点难以绕过 iOS 的当前限制。内置日历应用程序每天都会更改其图标以显示月份中的哪一天,我可以看到很多应用程序都可以从这种行为中受益 - 例如天气应用程序可以显示当前位置的天气状况图标。

  • 编辑:

    这是一个代码片段,用于取消任何现有的本地通知,并在 future 的午夜安排 64,并将正确的月份日期指定为角标(Badge)编号:
    [[UIApplication sharedApplication] cancelAllLocalNotifications];
    
    NSCalendar* calendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
    NSDateComponents* components = [calendar components:(NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit) fromDate:[NSDate date]];
    
    [components setHour:0];
    [components setMinute:0];
    [components setSecond:0];
    
    NSDate* midnightToday = [calendar dateFromComponents:components];
    
    const int SECONDS_PER_DAY = 60 * 60 * 24;
    
    for (int i = 1; i <= 64; i++)
    {
        NSDate* futureDate = [midnightToday dateByAddingTimeInterval:i * SECONDS_PER_DAY];
        NSDateComponents* components = [calendar components:NSDayCalendarUnit fromDate:futureDate];
    
        UILocalNotification* notification = [[UILocalNotification alloc] init];
        notification.fireDate = futureDate;
        notification.timeZone = [NSTimeZone systemTimeZone];
        notification.alertBody = nil;
        notification.soundName = nil;
        notification.applicationIconBadgeNumber = components.day;
    
        //NSLog(@"futureDate: %@", [NSDateFormatter localizedStringFromDate:futureDate dateStyle:NSDateFormatterMediumStyle timeStyle:NSDateFormatterMediumStyle]);
        //NSLog(@"notification: %@", notification);        
    
        [[UIApplication sharedApplication] scheduleLocalNotification:notification];
        [notification release];
    }
    
    [calendar release];
    

    这里发生的事情是:
  • 我们用 [NSDate date] 取今天的日期,然后把它分解成我们想要保留的组件(年、月、日),然后将时间组件(小时、分钟、秒)设置为午夜。这给了我们 midnightToday
  • 我们循环了 64 次,每次创建一个新的日期,即 future 1 到 64 天,方法是取 midnightToday 并加上每天的秒数乘以我们希望该日期在 future 的天数。我们使用此日期来安排本地通知。
  • 我们还需要计算出角标(Badge)编号的月份日期。因此,我们再次使用 NSDateComponents 将 future 日期分解为我们感兴趣的组件——在这种情况下只是日期,所以我们指定 NSDayCalendarUnit 。然后我们可以将 applicationIconBadgeNumber 设置为 components.day
  • 64 个通知中的每一个都与 UIApplication 一起安排,将在 future 1 到 64 天内交付。

  • 请注意,这可能仅适用于使用公历(西式)日历的用户 - 根据您的市场,您可能需要考虑世界各地使用的其他日历类型并支持这些日历类型。

    关于iphone - iOS 角标(Badge)编号实时更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8734078/

    相关文章:

    无论屏幕方向如何,iPhone 中心事件指示器

    ios - 带有 TableHeaderView 的 UITableView 不会在状态栏点击时滚动到顶部

    swift - 如何在 NSMenuItem 中显示搜索栏

    ios - 应用程序组和多个包 ID

    iphone - 使用 UIPanGestureRecognizer 拖动后必须点击两次才能选择 UITableViewCell

    iphone - 设置允许 airplay 不通过 airplay 发送声音

    iphone - 通用应用程序中的 EventKit (OS3.2)

    ios - 如何在导航栏的 Controller 之间实现更好的过渡?

    iphone - 适用于 iOS 应用程序的分析库

    iOS 8 从 PHAsset 快速加载图像