RORK LABEN
MAX — Rork MaxはWeb上でSwiftアプリを開発・公開できる初のビルダーをうたい、Xcode不要・2クリックでApp Store公開まで到達できますAPPLE — iPhone・iPad・Apple Watch・Apple TV・Vision Pro向けのネイティブSwiftアプリを生成しますEXPO — 通常版はReact Native(Expo)基盤。自然言語の要件記述からネイティブiOS/Androidアプリを生成しますFUNDING — Rorkがa16zから$2.8Mを調達。AIノーコードのモバイル領域で存在感を強めていますPRICE — 無料で開始でき、有料プランは$25/月から。個人開発者が試しやすい価格帯ですWWDC — WWDC 2026でApple Intelligenceが前進。ネイティブ機能の価値が上がり、ノーコード生成アプリのAI統合の選択肢も広がりますMAX — Rork MaxはWeb上でSwiftアプリを開発・公開できる初のビルダーをうたい、Xcode不要・2クリックでApp Store公開まで到達できますAPPLE — iPhone・iPad・Apple Watch・Apple TV・Vision Pro向けのネイティブSwiftアプリを生成しますEXPO — 通常版はReact Native(Expo)基盤。自然言語の要件記述からネイティブiOS/Androidアプリを生成しますFUNDING — Rorkがa16zから$2.8Mを調達。AIノーコードのモバイル領域で存在感を強めていますPRICE — 無料で開始でき、有料プランは$25/月から。個人開発者が試しやすい価格帯ですWWDC — WWDC 2026でApple Intelligenceが前進。ネイティブ機能の価値が上がり、ノーコード生成アプリのAI統合の選択肢も広がります
記事一覧/アプリ開発
アプリ開発/2026-06-14上級

WidgetKit のタイムラインを更新予算と向き合って設計する — 壁紙アプリのウィジェットが止まった理由

ホーム画面ウィジェットが「夕方になると更新されなくなる」のはなぜか。WidgetKit のタイムライン設計を、更新予算・リロードポリシー・エントリ密度の3点から整理しました。TimelineProvider の実コード、予算を使い切らないエントリ設計、relevance による優先度づけまで、個人開発で6本の壁紙アプリを運用しながら詰めた知見を共有します。

WidgetKit5TimelineProvideriOS76Rork Max147ウィジェット7パフォーマンス21

プレミアム記事

6本の壁紙アプリを個人開発で運用していて、ある時期から「ホーム画面ウィジェットの今日の一枚が、夕方以降は昨日のままになる」という報告が増えました。シミュレータでは正しく切り替わるのに、実機では夕方で止まる。この差を生んでいたのが、WidgetKit の更新予算という見えにくい仕組みでした。

ウィジェットはアプリと違い、好きなタイミングで好きなだけ描き直せるものではありません。システムが1日あたりの更新回数におおまかな予算を割り当てており、それを使い切ると、翌日まで更新が来なくなります。この前提を知らずにタイムラインを組むと、午前中に予算を食い尽くして午後は沈黙する、という今回の症状になります。

タイムラインは「未来の予定表」である

WidgetKit の考え方の中心は、ウィジェットが「いま何を表示するか」を毎回問い合わせるのではなく、「これからしばらくの表示予定」をまとめて提出する点にあります。TimelineProvider が返す Timeline は、複数の TimelineEntry を時刻つきで並べた予定表です。

システムはこの予定表に従って、指定時刻になったら次のエントリへ自動で切り替えます。つまり、未来の表示を先に計算して束ねて渡せば、その間システムへの問い合わせ(=予算消費)は発生しません。ここを理解すると、設計の方向性が決まります。

1エントリずつ刻むと予算が枯れる

最初に私がやってしまったのが、1時間ごとに1エントリだけ返し、.atEnd で「終わったらまた取りに来て」と繰り返す実装でした。これは一見正しく動きますが、リロードのたびに予算を消費するため、変化の多い日には昼過ぎで予算が尽きます。

正しくは、1回のタイムライン生成で未来の複数エントリをまとめて返します。たとえば1日分の切り替えを24本のエントリとして先に計算し、それを1つの Timeline に詰めて返せば、その日のうちはほぼ追加の問い合わせなしで回ります。

import WidgetKit
import SwiftUI
 
struct WallpaperProvider: TimelineProvider {
    func placeholder(in context: Context) -> WallpaperEntry {
        WallpaperEntry(date: Date(), imageName: "placeholder")
    }
 
    func getSnapshot(in context: Context, completion: @escaping (WallpaperEntry) -> Void) {
        completion(WallpaperEntry(date: Date(), imageName: todaysImageName()))
    }
 
    func getTimeline(in context: Context, completion: @escaping (Timeline<WallpaperEntry>) -> Void) {
        var entries: [WallpaperEntry] = []
        let calendar = Calendar.current
        let now = Date()
 
        // 未来 12 本を 2 時間刻みでまとめて生成し、束ねて返す
        for hourOffset in stride(from: 0, to: 24, by: 2) {
            guard let entryDate = calendar.date(byAdding: .hour, value: hourOffset, to: now) else { continue }
            let name = imageName(for: entryDate)
            entries.append(WallpaperEntry(date: entryDate, imageName: name))
        }
 
        // 翌日の頭で一度だけ次のタイムラインを取りに来る
        let tomorrow = calendar.date(byAdding: .day, value: 1, to: now)!
        completion(Timeline(entries: entries, policy: .after(tomorrow)))
    }
}

この実装の肝は、for ループで未来のエントリをまとめて作っている点と、policy.after(tomorrow) にして「次に取りに来るのは明日でよい」と宣言している点です。これで1日あたりの問い合わせ回数が劇的に減り、夕方の沈黙が消えました。

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

この記事の続きを読む

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

この記事で得られること
1日の更新予算を使い切らないために、1エントリずつ刻むのではなく未来を束ねて返す TimelineProvider の実装を、動く Swift で持ち帰れる
.atEnd・.after・.never の3つのリロードポリシーをどの種類のウィジェットで選ぶか、判断基準を表で確認できる
「夕方に更新が止まる」「タップしても古い画像のまま」という個人開発で実際に起きた2つの症状の原因と直し方がわかる
Stripe による安全な決済 · いつでもキャンセル可能

この記事を購入する

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

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

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

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

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

関連記事

アプリ開発2026-05-30
iOSの壁紙アプリでサムネイル一覧のカクつきを消す — ImageIOダウンサンプリングの実装メモ
大量のサムネイルを並べるとスクロールがカクつき、メモリも膨らむ。原因はメインスレッドでのフルサイズ画像デコードでした。ImageIOによるダウンサンプリングとプリフェッチで、実測でメモリを約80%削減した過程をまとめます。
アプリ開発2026-05-26
Rork アプリに WidgetKit で iOS ウィジェットを後付けする実装ノート — 壁紙アプリ 6 本に同時導入した設定とトレードオフ
Rork で骨格を作った iOS アプリに WidgetKit Extension を後付けする実装手順と、壁紙アプリ 6 本へ同時導入したときの App Group・Timeline Provider・メモリ上限まわりの実運用メモを、個人開発の現場目線でまとめます。
開発ツール2026-06-12
Rork 製アプリにホーム画面ウィジェットを追加する — Expo の制約を越えて WidgetKit を動かすまで
Rork が生成する Expo アプリにはホーム画面ウィジェットを直接追加できません。config plugin と App Group を使って WidgetKit を組み込む実装ルートを、詰まった点も含めて手順で整理します。
📚RECOMMENDED BOOKS
大規模言語モデル入門
山田育矢
LLM開発
生成AIプロンプトエンジニアリング入門
我妻幸長
プロンプト
Claude CodeによるAI駆動開発入門
平川知秀
AI駆動開発
※ アフィリエイトリンクを含みます
もっと見る →