RORK LABJP
MAX — Rork Max generates native Swift apps across iPhone, iPad, Watch, TV, Vision Pro, and iMessageNATIVE — It reaches AR/LiDAR scanning, Metal 3D games, widgets, Live Activities, and on-device Core MLFUNDING — Rork raised $2.8M from a16z, with 743K monthly visits and 85% growthPRICING — It's free to start, with paid plans beginning at $25 per monthFLOW — Describe your idea in plain English to get working code, a shareable test link, and iOS/Android buildsCOMPARE — The original Rork builds cross-platform apps on Expo/React Native; choose the right tool per goalMAX — Rork Max generates native Swift apps across iPhone, iPad, Watch, TV, Vision Pro, and iMessageNATIVE — It reaches AR/LiDAR scanning, Metal 3D games, widgets, Live Activities, and on-device Core MLFUNDING — Rork raised $2.8M from a16z, with 743K monthly visits and 85% growthPRICING — It's free to start, with paid plans beginning at $25 per monthFLOW — Describe your idea in plain English to get working code, a shareable test link, and iOS/Android buildsCOMPARE — The original Rork builds cross-platform apps on Expo/React Native; choose the right tool per goal
Articles/App Dev
App Dev/2026-06-27Intermediate

Show the In-App 'What's New' Once, Without Nagging — Version Gating and Seen State

An in-app 'What's New' screen that fires on fresh installs or shows every launch gets users annoyed. Here is a version-gated, seen-state design that delivers it exactly once to people who updated — in real Rork (Expo) code, plus a shape you reuse across multiple apps.

Rork461Expo114Onboarding8Retention design2Release ops

Premium Article

I have hit the wall of "nobody notices the new feature" many times. The store listing's "What's New in this version" goes mostly unread, and push notifications never reach people who declined them. The one reliable touchpoint left is the screen a user sees the moment they first open the app after updating.

But this "What's New" screen turns sour fast if you build it wrong. Showing a "new feature" announcement to someone who just installed, or showing the same notice every launch, only damages the experience. This is how to draw the boundaries so that "to people who updated, exactly once" actually holds.

How to decide "only people who updated"

The decision needs two values: the "current version" running now, and "which version this user last had." Persist the latter every time and compare it on launch.

  • No stored value → fresh install. Do not show.
  • Stored < current → updated. A candidate to show.
  • Stored = current → just relaunched on the same version. Do not show.

Read the current version from expo-application rather than threading app.json's version around by hand.

// lib/appVersion.ts
import * as Application from "expo-application";
 
// Display version like "1.4.0". It can be null, so keep a fallback.
export const CURRENT_VERSION = Application.nativeApplicationVersion ?? "0.0.0";
 
// A plain semver-ish compare of dotted version strings.
export function isNewer(a: string, b: string): boolean {
  const pa = a.split(".").map((n) => parseInt(n, 10) || 0);
  const pb = b.split(".").map((n) => parseInt(n, 10) || 0);
  for (let i = 0; i < Math.max(pa.length, pb.length); i++) {
    const x = pa[i] ?? 0;
    const y = pb[i] ?? 0;
    if (x !== y) return x > y;
  }
  return false;
}

Put the record and the decision in one place

Keep the decision in one small function. The important move is to separate "should we show What's New" from "should we update the last version." Persisting the last version happens unconditionally, whether or not anything was shown. Otherwise an update that has no What's New data yet will later batch two versions of notes into the next real update.

// lib/whatsNew.ts
import AsyncStorage from "@react-native-async-storage/async-storage";
import { CURRENT_VERSION, isNewer } from "./appVersion";
 
const LAST_VERSION_KEY = "whatsNew.lastVersion";
 
export type WhatsNewDecision =
  | { show: false }
  | { show: true; fromVersion: string; toVersion: string };
 
export async function evaluateWhatsNew(): Promise<WhatsNewDecision> {
  const last = await AsyncStorage.getItem(LAST_VERSION_KEY);
 
  // Always advance the last version (settle the bookkeeping first).
  const persist = () =>
    AsyncStorage.setItem(LAST_VERSION_KEY, CURRENT_VERSION);
 
  // Fresh install: record only, show nothing.
  if (!last) {
    await persist();
    return { show: false };
  }
 
  // Same, or current somehow older: do not show.
  if (!isNewer(CURRENT_VERSION, last)) {
    await persist();
    return { show: false };
  }
 
  // Updated: a candidate. Hand fromVersion to the caller.
  const decision: WhatsNewDecision = {
    show: true,
    fromVersion: last,
    toVersion: CURRENT_VERSION,
  };
  await persist();
  return decision;
}

The crux is calling persist() and returning show: false on a fresh install. The first run quietly records only, and from the next update onward you show correctly, exactly once.

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
A 'last version seen' record and decision logic that never shows on a fresh install, only once to users who updated
Reading the current version via expo-application and keying seen-state per version so nothing slips through
Running six wallpaper apps in parallel, a shared design that keeps the logic common and only the copy per app
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

App Dev2026-06-27
When a User Rewinds the Clock, Today's Card Shouldn't Break — Day Boundaries and Streak Integrity in a Daily-Content App
Daily 'card of the day' content breaks under timezone travel and manual clock changes, showing duplicates, gaps, or lost streaks. Here is a deterministic day-key and monotonic-clock design, in real Rork (Expo) code, that keeps it solid.
App Dev2026-06-03
Unifying Onboarding Across Six Wallpaper Apps: What One Month of First-Day Retention Showed Me
I folded the onboarding flows of six wallpaper apps scaffolded with Rork into a single config-driven component and watched first-day retention and push opt-in for a month. Here is an honest, operational note on what moved and what didn't.
App Dev2026-06-27
Before You Ask 'Are You Sure?' — Consider an Undoable Delete
Showing a confirmation dialog every time someone removes a list item trains them to tap OK without reading. Here is how to build an undoable delete in a Rork (Expo) app, and where confirmation dialogs still belong.
📚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 →