gpt4 book ai didi

ios - 后台根据时间激活本地通知

转载 作者:塔克拉玛干 更新时间:2023-11-02 10:10:48 24 4
gpt4 key购买 nike

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

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

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

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

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

添加:

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

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");
}

最佳答案

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

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

  1. 将您的控件连接到 View Controller header 中的 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. 在 View Controller 的实现文件中创建 IBAction 方法并连接每个控件(两个 UIDatePickers、一个 UISwitch 和一个 UIPickerView)。每个方法都应调用私有(private)方法 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 私有(private)方法中,我们(重新)启动了一个 NSTimer。我们在这里使用计时器,这样如果用户更改日期或快速切换开关(他们很可能会这样做),您就不会连续快速地拆除和设置通知。 (NSTimer 属性 userOptionInteractionTimer 应该在实现文件的接口(interface)延续中声明)。

    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. 创建另一个私有(private)方法来拆除预先存在的通知并设置新通知。

    在此处设置通知取决于您想要通知用户的时长和频率。假设您希望每小时通知一次用户,并且用户启用了 sleep 功能,那么您可以设置每小时 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. 请记住为您的所有日期选择器 View 和所有 IBAction 连接委托(delegate) IBAction,以及为您的选择器 View 连接委托(delegate)和数据源。还请记住设置您的委托(delegate)和数据源方法,以便填充选择器 View 。

  7. 就是这样!

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

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

关于ios - 后台根据时间激活本地通知,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15773544/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com