RORK LABEN
MAX — Rork MaxはReact NativeではなくネイティブSwiftアプリを生成。iPhone・iPad・Watch・TV・Vision Pro・iMessageに対応しますNATIVE — AR/LiDAR・Metalの3Dゲーム・Dynamic Island・Live Activities・HealthKit・Core MLなどネイティブ機能を解放しますCORE — 通常のRorkはReact Native(Expo)でiOS/Androidアプリを生成。自然言語からストア公開まで到達できますFUNDING — Rorkはa16zから$2.8Mを調達しましたGROWTH — 月間743,000訪問・成長率85%と勢いを増していますPRICING — 無料で始められ、有料プランは$25/月〜ですMAX — Rork MaxはReact NativeではなくネイティブSwiftアプリを生成。iPhone・iPad・Watch・TV・Vision Pro・iMessageに対応しますNATIVE — AR/LiDAR・Metalの3Dゲーム・Dynamic Island・Live Activities・HealthKit・Core MLなどネイティブ機能を解放しますCORE — 通常のRorkはReact Native(Expo)でiOS/Androidアプリを生成。自然言語からストア公開まで到達できますFUNDING — Rorkはa16zから$2.8Mを調達しましたGROWTH — 月間743,000訪問・成長率85%と勢いを増していますPRICING — 無料で始められ、有料プランは$25/月〜です
記事一覧/開発ツール
開発ツール/2026-06-23上級

アプリスイッチャーに個人的な画面が残る——背景へ回した瞬間のスナップショットを隠す設計

Rork が生成した React Native アプリをホームに戻すと、iOS はその瞬間の画面をアプリスイッチャー用に撮影してディスクに保存します。手記や個人的な入力画面はそこに残り、第三者の目に触れます。iOS の目隠しオーバーレイ(inactive で出す理由)、Android の FLAG_SECURE、機密画面だけに絞る方法、スクショ検知までを動くコードで示します。

Rork440React Native176Expo94プライバシー7セキュリティ設計

プレミアム記事

ある朝、自分の手記アプリを開いたまま電話に出ようとして、ホームボタンを二度押ししました。アプリスイッチャーに並んだサムネイルの中に、さっきまで書いていた個人的な一文がそのまま写っていました。家族に画面を覗かれて困るような内容ではありませんでしたが、「これは本人以外の手に渡るのを前提にしていなかった画面だ」と気づいて、少し背筋が冷えました。

iOS はアプリをホームへ戻した瞬間、その時点の画面を撮影してアプリスイッチャーのプレビューに使います。この画像はメモリ上だけのものではなく、ディスクに保存されます。つまり、パスコードでロックする前のホーム画面に、あなたのアプリの中身がしばらく残り続けるということです。手記、引き寄せの記録、健康データ、決済直前の金額——本人だけが見る前提で作った画面ほど、ここに残ると困ります。

私自身、個人開発で手記系・引き寄せ系のアプリを複数運用していますが、この「背景に回した瞬間」の守りは、Rork が生成した React Native のコードにも、Rork Max が書き出した Swift にも、初期状態では入っていません。守りの入り口は、その瞬間に画面へ目隠しを差し込むことにあります。つまずきやすい順序の問題から順に組み立てていきます。

守るべきは「表示している間」ではなく「離れる瞬間」

セキュリティ対策というと、画面を表示している最中の話だと考えがちです。けれども今回守りたいのは、ユーザーがアプリから離れる遷移の一瞬です。iOS はアプリが前面から外れるとき、UI のスナップショットを撮ってアプリスイッチャーに使います。撮影されるのは「完全に背景へ落ちた後」ではなく、その手前の inactive 状態に入るタイミングです。

React Native の AppState は、この遷移を3つの値で表します。前面で動いている active、操作を受け付けない過渡状態の inactive(iOS のみ)、そして背景の background です。コントロールセンターを下ろした、電話がかかってきた、アプリスイッチャーを開いた——こうした場面ではまず inactive を通過します。

ここに最初の落とし穴があります。background になってから目隠しを出すと、スナップショットの撮影に間に合いません。撮影はすでに inactive の段階で済んでいるからです。守りの入り口は background ではなく inactive だ、という一点を最初に押さえておきます。

iOS:inactive で目隠しオーバーレイを出す

まず、アプリのルートに常駐する目隠し用のオーバーレイを用意します。active のときは透明(実質非表示)、それ以外のときは画面全体を覆う、という素直な作りです。状態管理は外部ライブラリを使わず、軽量な AppState 購読フックにまとめます。

// hooks/usePrivacyShield.ts
import { useEffect, useRef, useState } from "react";
import { AppState, AppStateStatus } from "react-native";
 
// active 以外(inactive / background)のときに true を返す。
// iOS のスナップショットは inactive で撮られるため、background を待ってはいけない。
export function usePrivacyShield(enabled: boolean) {
  const [shielded, setShielded] = useState(false);
  const appState = useRef<AppStateStatus>(AppState.currentState);
 
  useEffect(() => {
    if (!enabled) {
      setShielded(false);
      return;
    }
 
    const handle = (next: AppStateStatus) => {
      // active へ戻った瞬間だけ目隠しを外す。
      // それ以外の遷移(inactive / background)はすべて覆う側に倒す。
      setShielded(next !== "active");
      appState.current = next;
    };
 
    // 初期表示時の取りこぼし防止(起動直後が inactive のことがある)。
    setShielded(AppState.currentState !== "active");
 
    const sub = AppState.addEventListener("change", handle);
    return () => sub.remove();
  }, [enabled]);
 
  return shielded;
}

このフックをアプリ最上位の1か所で使い、覆う UI を被せます。画面ごとに置くと、遷移の瞬間に「今どの画面か」を判定する余裕がないため、必ずルートに置きます。

// components/PrivacyShield.tsx
import { View, StyleSheet, Image } from "react-native";
import { usePrivacyShield } from "../hooks/usePrivacyShield";
 
export function PrivacyShield({ enabled }: { enabled: boolean }) {
  const shielded = usePrivacyShield(enabled);
  if (!shielded) return null;
 
  // ブランドのロゴ1枚だけを中央に置く。中身は一切描かない。
  return (
    <View style={styles.cover} pointerEvents="none">
      <Image
        source={require("../assets/shield-logo.png")}
        style={{ width: 96, height: 96, opacity: 0.9 }}
        resizeMode="contain"
      />
    </View>
  );
}
 
const styles = StyleSheet.create({
  cover: {
    ...StyleSheet.absoluteFillObject,
    backgroundColor: "#0E1116",
    alignItems: "center",
    justifyContent: "center",
    zIndex: 9999,
  },
});

ルートではこう組み込みます。enabled には「いま機密画面にいるか」を渡せるようにしておくと、後で画面単位の制御に発展させられます。

// app/_layout.tsx(Expo Router の例)
import { Stack } from "expo-router";
import { PrivacyShield } from "../components/PrivacyShield";
 
export default function RootLayout() {
  return (
    <>
      <Stack />
      {/* まずはアプリ全体で常に有効化。後で画面単位に絞る */}
      <PrivacyShield enabled={true} />
    </>
  );
}

ここまでで、ホームに戻したりアプリスイッチャーを開いたりした瞬間、画面はロゴ1枚の目隠しに切り替わり、撮影されるスナップショットも目隠し後の状態になります。

ここまでお読みいただきありがとうございます。

この記事の続きを読む

この先には、実装コードやベンチマーク結果など、実務でお役に立てる内容をご用意しています。このサイトは広告を掲載しておらず、サーバーや開発にかかる費用はメンバーの皆様のご支援で成り立っています。もしお役に立てていましたら、ご支援いただけますと大変ありがたいです。

この記事で得られること
個人的な入力画面がアプリスイッチャーに残っていた、という不安を抱えていた人が、背景遷移の瞬間に目隠しを差し込む実装を今日手に入れられる
なぜ background では遅れて inactive で出す必要があるのか、iOS と Android で守り方がまったく違う理由を理解し、片方だけ守って安心する事故を避けられる
アプリ全体ではなく機密画面だけに FLAG_SECURE を絞り、スクリーンショット検知でユーザーに知らせる本番運用パターンを実装できる
Stripe による安全な決済 · いつでもキャンセル可能

この記事を購入する

この先の内容をすべてお読みいただけます。一度のご購入で、いつでも何度でもアクセスできます。このサイトは広告を掲載しておらず、皆さまのご支援がサーバー費用などの運営を支えています。

または
メンバーシップなら全記事が読み放題 →
シェア

お読みいただきありがとうございます

Rork Lab は広告なしで運営しており、サーバー費用などの運営コストはメンバーシップのご支援で賄っています。実装コード・ベンチマーク・本番設計パターンなど、実務でお役立ていただける記事を毎日更新しています。もし読んでよかったと感じていただけましたら、ぜひご覧ください。

  • コピー&ペーストで使える実装コード付き
  • 毎日新しい上級ガイドを追加
  • ¥580/月 または ¥1,480 の永久アクセス
メンバーシップを見る →

関連記事

開発ツール2026-06-22
壊れたキャッシュで毎回起動時に落ちるアプリ——ユーザーが自力で抜け出せる「セーフモード起動」を設計する
永続化したキャッシュが壊れて起動のたびに同じ場所で落ちると、ユーザーには再インストールしか残りません。アプリ自身が早期クラッシュの連続を数え、対話可能になって初めて起動を確定し、危険な状態だけを段階的にリセットするセーフモードを、Expo(React Native)の実装で設計します。
開発ツール2026-06-22
Rork(Expo)のトーストは、重なっても読み上げられても崩れない設計にする
Rork が生成した React Native アプリにトースト通知を足すとき、単純な実装は『同時に2つ出ると重なる』『スクリーンリーダーに無視される』『ノッチやホームインジケータに隠れる』の3点で破綻します。ルート1か所に置くキュー設計、本体の再レンダーから切り離すアニメーション、AccessibilityInfo による読み上げ、セーフエリア対応までを動くコードで示します。
開発ツール2026-06-21
Rorkアプリの写真が、アップロードした先でだけ横向きになる——EXIF Orientation の正規化と位置情報メタデータの削除
端末では正しく見える写真が、サーバーや他端末でだけ90度回転する原因はEXIF Orientationです。expo-image-manipulatorで回転を焼き込み、位置情報メタデータを削除してからアップロードする実装を解説します。
📚RECOMMENDED BOOKS
大規模言語モデル入門
山田育矢
LLM開発
生成AIプロンプトエンジニアリング入門
我妻幸長
プロンプト
Claude CodeによるAI駆動開発入門
平川知秀
AI駆動開発
※ アフィリエイトリンクを含みます
もっと見る →