|
| 1 | +# ホットリロード |
| 2 | + |
| 3 | +FrankenPHPには、開発者のエクスペリエンスを大幅に向上させるために設計された組み込みの**ホットリロード**機能が含まれています。 |
| 4 | + |
| 5 | + |
| 6 | + |
| 7 | +この機能は、ViteやwebpackなどのモダンなJavaScriptツールにおける**ホットモジュールリプレースメント (HMR)** に似たワークフローを提供します。 |
| 8 | +ファイルの変更(PHPコード、テンプレート、JavaScript、CSSファイルなど)のたびに手動でブラウザをリフレッシュする代わりに、 |
| 9 | +FrankenPHPはページの内容をリアルタイムで更新します。 |
| 10 | + |
| 11 | +ホットリロードは、WordPress、Laravel、Symfony、その他すべてのPHPアプリケーションやフレームワークでネイティブに動作します。 |
| 12 | + |
| 13 | +有効にすると、FrankenPHPは現在の作業ディレクトリにおけるファイルシステムの変更を監視します。 |
| 14 | +ファイルが変更されると、[Mercure](mercure.md)の更新をブラウザにプッシュします。 |
| 15 | + |
| 16 | +設定に応じて、ブラウザは以下のいずれかを実行します。 |
| 17 | + |
| 18 | +- [Idiomorph](https://github.com/bigskysoftware/idiomorph)がロードされている場合、**DOMをモーフィング**します(スクロール位置と入力状態を保持)。 |
| 19 | +- Idiomorphが存在しない場合、**ページをリロード**します(標準のライブリロード)。 |
| 20 | + |
| 21 | +## 設定 |
| 22 | + |
| 23 | +ホットリロードを有効にするには、Mercureを有効にしてから、`Caddyfile`の`php_server`ディレクティブに`hot_reload`サブディレクティブを追加します。 |
| 24 | + |
| 25 | +> [!WARNING] |
| 26 | +> |
| 27 | +> この機能は**開発環境のみ**を対象としています。 |
| 28 | +> `hot_reload`を本番環境で有効にしないでください。この機能は安全ではなく(機密性の高い内部詳細を公開します)、アプリケーションの速度を低下させます。 |
| 29 | +> |
| 30 | +```caddyfile |
| 31 | +localhost |
| 32 | +
|
| 33 | +mercure { |
| 34 | + anonymous |
| 35 | +} |
| 36 | +
|
| 37 | +root public/ |
| 38 | +php_server { |
| 39 | + hot_reload |
| 40 | +} |
| 41 | +``` |
| 42 | + |
| 43 | +デフォルトでは、FrankenPHPは現在の作業ディレクトリ内の以下のグロブパターンに一致するすべてのファイルを監視します: `./**/*.{css,env,gif,htm,html,jpg,jpeg,js,mjs,php,png,svg,twig,webp,xml,yaml,yml}` |
| 44 | + |
| 45 | +グロブ構文を使用して、監視するファイルを明示的に設定することも可能です。 |
| 46 | + |
| 47 | +```caddyfile |
| 48 | +localhost |
| 49 | +
|
| 50 | +mercure { |
| 51 | + anonymous |
| 52 | +} |
| 53 | +
|
| 54 | +root public/ |
| 55 | +php_server { |
| 56 | + hot_reload src/**/*{.php,.js} config/**/*.yaml |
| 57 | +} |
| 58 | +``` |
| 59 | + |
| 60 | +使用するMercureトピック、および監視するディレクトリまたはファイルを指定するには、`hot_reload`のロングフォームを使用します。 |
| 61 | + |
| 62 | +```caddyfile |
| 63 | +localhost |
| 64 | +
|
| 65 | +mercure { |
| 66 | + anonymous |
| 67 | +} |
| 68 | +
|
| 69 | +root public/ |
| 70 | +php_server { |
| 71 | + hot_reload { |
| 72 | + topic hot-reload-topic |
| 73 | + watch src/**/*.php |
| 74 | + watch assets/**/*.{ts,json} |
| 75 | + watch templates/ |
| 76 | + watch public/css/ |
| 77 | + } |
| 78 | +} |
| 79 | +``` |
| 80 | + |
| 81 | +## クライアントサイドの統合 |
| 82 | + |
| 83 | +サーバーは変更を検出しますが、ブラウザはこれらのイベントを購読してページを更新する必要があります。 |
| 84 | +FrankenPHPは、ファイル変更を購読するために使用するMercure HubのURLを、`$_SERVER['FRANKENPHP_HOT_RELOAD']`環境変数を通じて公開します。 |
| 85 | + |
| 86 | +クライアントサイドのロジックを処理するための便利なJavaScriptライブラリ、[frankenphp-hot-reload](https://www.npmjs.com/package/frankenphp-hot-reload)も利用可能です。 |
| 87 | +これを使用するには、メインレイアウトに以下を追加します。 |
| 88 | + |
| 89 | +```php |
| 90 | +<!DOCTYPE html> |
| 91 | +<title>FrankenPHP Hot Reload</title> |
| 92 | +<?php if (isset($_SERVER['FRANKENPHP_HOT_RELOAD'])): ?> |
| 93 | +<meta name="frankenphp-hot-reload:url" content="<?=$_SERVER['FRANKENPHP_HOT_RELOAD']?>"> |
| 94 | +<script src="https://cdn.jsdelivr.net/npm/idiomorph"></script> |
| 95 | +<script src="https://cdn.jsdelivr.net/npm/frankenphp-hot-reload/+esm" type="module"></script> |
| 96 | +<?php endif ?> |
| 97 | +``` |
| 98 | + |
| 99 | +このライブラリは自動的にMercureハブを購読し、ファイル変更が検出されるとバックグラウンドで現在のURLをフェッチし、DOMをモーフィングします。 |
| 100 | +[npm](https://www.npmjs.com/package/frankenphp-hot-reload)パッケージとして、また[GitHub](https://github.com/dunglas/frankenphp-hot-reload)で利用できます。 |
| 101 | + |
| 102 | +または、`EventSource`ネイティブJavaScriptクラスを使用してMercureハブに直接購読することで、独自のクライアントサイドロジックを実装することもできます。 |
| 103 | + |
| 104 | +### 既存のDOMノードを保持する |
| 105 | + |
| 106 | +まれに、[Symfonyのウェブデバッグツールバーなどの開発ツール](https://github.com/symfony/symfony/pull/62970)を使用している場合など、特定のDOMノードを保持したいことがあります。 |
| 107 | +そのためには、関連するHTML要素に`data-frankenphp-hot-reload-preserve`属性を追加します。 |
| 108 | + |
| 109 | +```html |
| 110 | +<div data-frankenphp-hot-reload-preserve><!-- My debug bar --></div> |
| 111 | +``` |
| 112 | + |
| 113 | +## ワーカーモード |
| 114 | + |
| 115 | +アプリケーションを[ワーカーモード](https://frankenphp.dev/docs/worker/)で実行している場合、アプリケーションスクリプトはメモリに常駐します。 |
| 116 | +これは、ブラウザがリロードされても、PHPコードの変更がすぐに反映されないことを意味します。 |
| 117 | + |
| 118 | +最高の開発者エクスペリエンスのためには、`hot_reload`を[ワーカーディレクティブ内の`watch`サブディレクティブ](config.md#watching-for-file-changes)と組み合わせるべきです。 |
| 119 | + |
| 120 | +- `hot_reload`: ファイルが変更されたときに**ブラウザ**をリフレッシュします |
| 121 | +- `worker.watch`: ファイルが変更されたときにワーカーを再起動します |
| 122 | + |
| 123 | +```caddy |
| 124 | +localhost |
| 125 | +
|
| 126 | +mercure { |
| 127 | + anonymous |
| 128 | +} |
| 129 | +
|
| 130 | +root public/ |
| 131 | +php_server { |
| 132 | + hot_reload |
| 133 | + worker { |
| 134 | + file /path/to/my_worker.php |
| 135 | + watch |
| 136 | + } |
| 137 | +} |
| 138 | +``` |
| 139 | + |
| 140 | +## 仕組み |
| 141 | + |
| 142 | +1. **監視**: FrankenPHPは、内部で[`e-dant/watcher`ライブラリ](https://github.com/e-dant/watcher)を使用してファイルシステムの変更を監視します(私たちはGoバインディングに貢献しました)。 |
| 143 | +2. **再起動 (ワーカーモード)**: ワーカー設定で`watch`が有効になっている場合、新しいコードをロードするためにPHPワーカーが再起動されます。 |
| 144 | +3. **プッシュ**: 変更されたファイルのリストを含むJSONペイロードが、組み込みの[Mercureハブ](https://mercure.rocks)に送信されます。 |
| 145 | +4. **受信**: JavaScriptライブラリを介してリッスンしているブラウザがMercureイベントを受信します。 |
| 146 | +5. **更新**: |
| 147 | + |
| 148 | + - **Idiomorph**が検出された場合、更新されたコンテンツをフェッチし、現在のHTMLを新しい状態に合わせてモーフィングし、状態を失うことなく即座に変更を適用します。 |
| 149 | + - それ以外の場合、`window.location.reload()`が呼び出されてページがリフレッシュされます。 |
0 commit comments