RORK LABJP
MAX — Rork Max builds native Swift apps instead of React Native, supporting iPhone, iPad, Watch, TV, Vision Pro, and iMessageNATIVE — It unlocks native capabilities: AR/LiDAR, Metal 3D games, Dynamic Island, Live Activities, HealthKit, and Core MLCORE — Standard Rork generates iOS/Android apps with React Native (Expo), taking you from plain English to the app storesFUNDING — Rork raised $2.8M from a16z (Andreessen Horowitz)GROWTH — The platform now sees 743,000 monthly visits with 85% growthPRICING — Free to start, with paid plans from $25/monthMAX — Rork Max builds native Swift apps instead of React Native, supporting iPhone, iPad, Watch, TV, Vision Pro, and iMessageNATIVE — It unlocks native capabilities: AR/LiDAR, Metal 3D games, Dynamic Island, Live Activities, HealthKit, and Core MLCORE — Standard Rork generates iOS/Android apps with React Native (Expo), taking you from plain English to the app storesFUNDING — Rork raised $2.8M from a16z (Andreessen Horowitz)GROWTH — The platform now sees 743,000 monthly visits with 85% growthPRICING — Free to start, with paid plans from $25/month
Articles/Dev Tools
Dev Tools/2026-06-22Advanced

Your Daily Reminder Stops Firing After a Couple of Weeks — iOS's Invisible 64-Notification Cap

When a daily reminder built with Rork (Expo) goes silent after a while, the cause is usually iOS's 64 pending-notification limit. Design a repeating calendar trigger for fixed messages and a rolling reschedule for daily-changing content, with working code that survives DST and multiple reminders.

Rork440expo-notifications5Local NotificationsReact Native175iOS83

Premium Article

"Deliver one line every morning at 8." When I added that to a daily affirmation app, I naively scheduled thirty days' worth in one shot. It worked perfectly in the simulator. Then, a while after release, users started writing in: "It fired for the first few days, then stopped."

It wasn't a bug in my code. It was a limit the official tutorials barely mention: iOS only holds up to 64 pending local notifications per app. Thirty days is far over that — and if you run a morning-and-evening pair, you can't even bank a full month. Anything past the cap is dropped silently, with no error.

This article walks through rebuilding a daily reminder so it keeps firing, using working expo-notifications code. I'll leave the basic setup to the local notifications setup guide and focus only on living within the cap.

iOS keeps only 64 pending notifications — the rest are dropped silently

Apple keeps at most 64 un-fired (pending) local notifications per app. If you try to schedule a 65th with scheduleNotificationAsync, no exception is thrown. iOS keeps the 64 with the soonest fire times and quietly drops the rest from the pending list.

The subtle part: a repeats: true notification counts as one slot, no matter how many times it will fire. "Every day at 8" built as a repeating trigger costs a single slot forever. But the moment you want "a different body every day," repeats is off the table, and you start banking one notification per date — which hits the 64-slot wall fast.

So the decision is simple:

  • The text can be the same every day → one repeating trigger. The cap is basically a non-issue.
  • You want the text to change daily → individual scheduling, and you must budget the 64 slots.

I lost time early on by not making this call up front. Classifying each reminder as "fixed text" or "daily-changing" before writing anything makes the whole design fall into place.

Measure first — how many are you actually banking?

Before changing the design, measure. getAllScheduledNotificationsAsync returns the current pending array, so a single log line at launch shows how many slots you're using.

import * as Notifications from 'expo-notifications';
 
// Call once at launch. Guard with __DEV__ in production to keep logs quiet.
export async function inspectPendingNotifications(): Promise<number> {
  const pending = await Notifications.getAllScheduledNotificationsAsync();
  if (__DEV__) {
    console.log(`[notif] pending=${pending.length}/64`);
    pending.forEach((n) => {
      // Trigger shape differs by platform, so log it loosely.
      console.log('  -', n.identifier, JSON.stringify(n.trigger));
    });
  }
  return pending.length;
}

If pending.length ever crosses 60, you're already in the danger zone. In my production apps I ship this number to internal metrics on every launch, so I never discover after the fact that I've been pinned at the cap. Banking notifications without watching the count is like topping up fuel without looking at the gauge.

Thank you for reading this far.

Continue Reading

What follows includes implementation code, benchmarks, and practical content we hope you'll find useful. This site runs without ads — server and development costs are supported entirely by members like you. If it's been helpful, we'd be truly grateful for your support.

WHAT YOU'LL LEARN
You'll understand why a reminder that delivers a different line each morning suddenly goes quiet — iOS's invisible 64 pending-notification cap — and how to design around it
You'll be able to decide, from working code, when a single repeating calendar trigger is enough and when daily-changing content needs a rolling reschedule
You'll be able to rebuild your own app so the fire time never drifts across DST, time-zone moves, or a morning-plus-evening reminder pair
Secure payment via Stripe · Cancel anytime

Unlock This Article

Get full access to the rest of this article. Buy once, read anytime. This site is ad-free — your support goes directly toward keeping it running.

or
Unlock all articles with Membership →
Share

Thank You for Reading

Rork Lab is ad-free, supported entirely by members like you. We publish practical guides daily with implementation code, benchmarks, and production-ready patterns. If you've found it useful, we'd love to have you on board.

  • Copy-paste ready implementation code
  • New advanced guides published daily
  • $5/mo or $10 for lifetime access
View Membership →

Related Articles

Dev Tools2026-05-28
Tracking Down BGTaskScheduler.submit Error Code=1 (Unavailable) in Rork iOS Apps
A field-tested checklist for diagnosing BGTaskScheduler.submit failing with Error Code=1 (Unavailable) in iOS apps built with Rork, walking through the six causes that account for nearly every case.
Dev Tools2026-05-26
iOS Foreground Notifications Disappear in Expo — A Practical Fix Guide
A focused walkthrough of why iOS silently drops push notifications when the app is in the foreground, and how to wire setNotificationHandler correctly for iOS 14+ with expo-notifications.
Dev Tools2026-05-18
Fixing iOS-Only Linking.canOpenURL False Returns in Rork — The LSApplicationQueriesSchemes Trap
Production iOS builds silently route every user to the mobile web instead of opening TikTok or X? This is almost always a missing LSApplicationQueriesSchemes entry. Here is the full fix path for Rork.
📚RECOMMENDED BOOKS
Build a Large Language Model (From Scratch)
Sebastian Raschka
LLM Dev
Prompt Engineering for LLMs
Berryman & Ziegler
Prompting
AI Engineering
Chip Huyen
AI Eng
* Contains affiliate links
See all →