2

所以我有一个包含重复间隔本地通知的应用程序,我想添加一个在睡眠时间暂停通知的功能。

到目前为止,我已经为用户创建了两个日期选择器来指定他们想要停止重复间隔的时间以及自动重新启动的时间。我为他们添加了一个 uiswitch 来激活睡眠模式或忽略该功能。

现在,我将如何让我的主要 uipickerview -(他们从这里选择通知)来收听 uiswitch(如果它打开),然后它在来自我的第一个 datepicker 的时间暂停通知并重新启动通知来自第二个日期选择器的时间?

我已经设置了我的日期选择器和我的 uiswitch,但不知道如何用我的 uipickerview 实现它。它应该在 DidSelectRow 的方法下吗?还是 appdelegate 中的方法(例如 DidEnterBackground)?

请询问您是否需要更多信息或代码来理解这个想法并帮助我。谢谢。

添加在:

这是我为日期选择器准备的代码,但是,我只是缺少将其正确添加到我的选择器视图的连接。

NSDateFormatter *dateFormatter = [[NSDateFormatter alloc]init];
dateFormatter.timeZone=[NSTimeZone defaultTimeZone];
dateFormatter.timeStyle=NSDateFormatterShortStyle;
dateFormatter.dateStyle=NSDateFormatterShortStyle;
NSString *dateTimeString=[dateFormatter stringFromDate:startTime.date];
NSLog(@"Start time is %@",dateTimeString);


NSDateFormatter *dateFormatter2 = [[NSDateFormatter alloc]init];
dateFormatter2.timeZone=[NSTimeZone defaultTimeZone];
dateFormatter2.timeStyle=NSDateFormatterShortStyle;
dateFormatter2.dateStyle=NSDateFormatterShortStyle;
NSString *dateTimeString2=[dateFormatter2 stringFromDate:endTime.date];
NSLog(@"End time is %@",dateTimeString2);

我也有这段代码来比较时间:

if ([[NSDate date] isEqualToDate:startTime.date]) {
NSLog(@"currentDate is equal to startTime"); 
}

if ([[NSDate date] isEqualToDate:endTime.date]) {
NSLog(@"currentDate is equal to endTime"); 
}
4

2 回答 2

2

iOS 不支持在两个特定日期/时间之间发布本地重复通知。此外,与基于地理的通知不同,当不在前台时,应用程序不会收到触发通知的警报。所以我们需要通过在用户没有睡着的时候创建许多单独的通知来解决这个问题。

请按照以下步骤创建基本解决方案:

  1. 将您的控件连接到视图控制器标题中的 IBOutlets:

    SomeViewController.h:

    @interface SomeViewController : UIViewController
    
    @property (weak, nonatomic) IBOutlet UISwitch *sleepToggleSwitch;
    @property (weak, nonatomic) IBOutlet UIDatePicker *notificationIgnoreStartTime;
    @property (weak, nonatomic) IBOutlet UIDatePicker *notificationIgnoreEndTime;
    @property (weak, nonatomic) IBOutlet UIPickerView *notificationTypePickerView;
    
    @end
    
  2. 在视图控制器的实现文件中创建 IBAction 方法并连接每个控件(两个 UIDatePickers、一个 UISwitch 和一个 UIPickerView)。每个方法都应该调用私有方法startUserOptionInteractionTimer

    SomeViewController.m:

    - (IBAction)noNotificationPeriodStartDateChanged:(id)sender
    {
        [self startUserOptionInteractionTimer];
    }
    
    - (IBAction)noNotificationPeriodEndDateChanged:(id)sender
    {
        [self startUserOptionInteractionTimer];
    }
    
    - (IBAction)sleepToggleSwitchToggled:(id)sender
    {
        [self startUserOptionInteractionTimer];
    }
    
    - (IBAction)notificationTypeChanged:(id)sender
    {
        [self startUserOptionInteractionTimer];
    }
    
  3. startUserOptionInteractionTimer私有方法中,我们(重新)启动一个 NSTimer。我们在这里使用了一个计时器,这样如果用户更改日期或快速切换开关(他们可能会这样做),您就不会连续快速地关闭和设置通知。(NSTimer 属性userOptionInteractionTimer应该在实现文件的接口延续中声明)。

    SomeViewController.m:

    - (void)startUserOptionInteractionTimer
    {
        // Remove any existing timer
        [self.userOptionInteractionTimer invalidate];
        self.userOptionInteractionTimer = [NSTimer scheduledTimerWithTimeInterval:4.f
                                                                           target:self
                                                                         selector:@selector(setupNotifications)
                                                                         userInfo:nil
                                                                          repeats:NO];
    }
    
  4. 创建另一个私有方法来拆除预先存在的通知并设置新的通知。

    在此处设置通知取决于您希望通知用户的时间和频率。假设您想每小时通知您的用户并且用户启用了睡眠功能,那么您将为每天重复的每小时设置 14-18 个通知(取决于用户睡眠时间)。

    SomeViewController.m:

    - (void)setupNotifications
    {
        [[UIApplication sharedApplication] cancelAllLocalNotifications];
    
        // Read the notification type from the notification type picker view
        NSInteger row = [self.notificationTypePickerView selectedRowInComponent:0]; // Assumes there is only one component in the picker view.
        NSString *notificationType = [self.notificationList objectAtIndex:row]; // Where notificationList is the array storing the list of notification strings that appear in the picker view.
    
        // If the user has turned the sleep feature on (via the UISwitch):
        if (self.sleepToggleSwitch.on) {
            // Set the first notification to start after the user selected 'noNotificationPeriodEndDate' and repeat daily.
            // Add 1 hour to the notification date
            // Do while the notification date < the user selected 'noNotificationPeriodStartDate' ...
            //     Create the notification and set to repeat daily
            //     Add 1 hour to the notification date
            // Loop
        } else {
            // Set up 24 repeating daily notifications each one hour apart.
        }
    }
    

    请记住,单个应用程序最多只能创建 64 个通知(重复通知计为一个),因此,如果您希望通知在几天或几周内的不同时间触发,您可能需要重新考虑您的设计小的。

  5. 在 NSUserDefaults 中加载和存储用户选择的首选项:

    SomeViewController.m:

    - (void)viewDidLoad
    {
        [super viewDidLoad];
    
        // Load the user defaults
        NSDate *sleepStartDate = [[NSUserDefaults standardUserDefaults] objectForKey:@"SleepStartDate"];
        self.notificationIgnoreStartTime.date = sleepStartDate ? sleepStartDate : [NSDate date];
        NSDate *sleepEndDate = [[NSUserDefaults standardUserDefaults] objectForKey:@"SleepEndDate"];
        self.notificationIgnoreEndTime.date = sleepEndDate ? sleepEndDate : [NSDate date];
        self.sleepToggleSwitch.on = [[NSUserDefaults standardUserDefaults] boolForKey:@"SleepEnabled"];
    
        // Watch for when the app leaves the foreground
        [[NSNotificationCenter defaultCenter] addObserver:self
                                                 selector:@selector(applicationWillResignActive)
                                                     name:UIApplicationWillResignActiveNotification object:nil];
    }
    
    - (void)applicationWillResignActive
    {
        // If the timer is still waiting, fire it so the notifications are setup correctly before the app enters the background.
        if (self.userOptionInteractionTimer.isValid)
            [self.userOptionInteractionTimer fire];
    
        // Store the user's selections in NSUserDefaults
        [[NSUserDefaults standardUserDefaults] setObject:self.notificationIgnoreStartTime.date forKey:@"SleepStartDate"];
        [[NSUserDefaults standardUserDefaults] setObject:self.notificationIgnoreEndTime.date forKey:@"SleepEndDate"];
        [[NSUserDefaults standardUserDefaults] setBool:self.sleepToggleSwitch.on forKey:@"SleepEnabled"];
    }
    

    另外,请注意,如果应用程序即将进入后台(即离开前台)并且计时器仍在计时,我们会强制它触发,以便在计时器被操作系统杀死之前设置通知。

  6. 请记住为您的所有日期选择器视图和所有 IBActions 以及您的选择器视图的委托和数据源连接代理 IBAction。还要记住设置您的委托和数据源方法,以便填充选择器视图。

  7. 就是这样!

上述设计将确保在正确的时间触发通知,无论应用程序是在前台、后台还是终止。(但是,如果在收到通知时应用程序在前台,用户将不会收到通知。相反,application:didReceiveLocalNotification:会调用 appDelegate)。

显然,上面的代码不是“准备好执行”,但你应该能够填补空白。

于 2013-04-11T09:56:48.060 回答
1

在 a或 the中UISwitch设置一个可以在任何地方检查的属性,可能会激活和停用 Picker(userInteractionEnabled 等)。plistNSUserDefaults

还连接了UISwitch一个IBAction方法,ValueChanged以便能够始终切换状态。

进一步使用只是更新时间的DidSelectRowAtIndexPath:方法UIPickerView(也可以将它们保存在一个plist或者NSUserDefaults如果您需要它们在多个地方。
或者您将属性用作类属性(静态)^^

对于自动检查,我将只使用在NSTimer60 秒内循环运行AppDelegate的事件,并触发一个事件来检查它是否可以继续或返回,或者如果你想触发它已经在这里检查(可以扩展以确保你正在触发事件不是在一分钟内的任何地方,而是在 00 秒)。
检查您是否可以开火的样本对于该州来说可能看起来像这样

startSilentTimePicker : 晚上 10:30
stopSilentTimePicker : 早上 7:15
activitySwitch : on

// filtered date by only time
NSDateComponents *startComps = [[NSCalendar currentCalendar] components:NSHourCalendarUnit|NSMinuteCalendarUnit fromDate:[startSilentTimePicker date]];
NSDateComponents *endComps = [[NSCalendar currentCalendar] components:NSHourCalendarUnit|NSMinuteCalendarUnit fromDate:[stopSilentTimePicker date]];
NSDateComponents *currentComps = [[NSCalendar currentCalendar] components:NSHourCalendarUnit|NSMinuteCalendarUnit fromDate:[NSDate date]];

if([[startComps date] laterDate:[currentComps date]] == [currentComps date] || 
[[endComps date] earlierDate:[currentComps date]] == [currentComps date])
{
    // set the value, that tells you beeing in silent time, set a return from the methode or invert to fire notification here
}

没有用我的代码实际测试过它,但我正在使用类似的东西。如果您遗漏了什么,请告诉我,也许我可以完成;)

于 2013-04-09T07:03:02.683 回答