From b98ebfa6de1ab2eaa07beb7ac6a5ecb26f75af0a Mon Sep 17 00:00:00 2001 From: <> Date: Wed, 18 Dec 2024 15:47:43 +0000 Subject: [PATCH] Deployed 2fdc7e3 with MkDocs version: 1.6.1 --- .nojekyll | 0 404.html | 1557 ++++ ai/attacker/index.html | 1674 ++++ ai/ball_possession/index.html | 2009 +++++ ai/coordinates/index.html | 1698 +++++ ai/diagram/index.html | 1841 +++++ ai/docker/index.html | 1781 +++++ ai/grSim/index.html | 1628 ++++ ai/index.html | 1620 ++++ ai/logs/2023-04-16/index.html | 1768 +++++ ai/logs/2023-05-03/index.html | 1876 +++++ ai/logs/2023-05-06/index.html | 2027 +++++ ai/logs/2023-05-07/index.html | 1979 +++++ ai/logs/2023-05-13/index.html | 1781 +++++ ai/logs/2023-05-14/index.html | 1823 +++++ ai/logs/2023-05-27/index.html | 1844 +++++ ai/logs/2023-05-28/index.html | 1832 +++++ ai/logs/2023-05-30/index.html | 1939 +++++ ai/logs/2023-06-06/index.html | 1811 +++++ ai/logs/2023-06-07/index.html | 1766 +++++ ai/logs/2023-06-12/index.html | 1755 +++++ ai/logs/2023-06-17/index.html | 1896 +++++ ai/logs/2023-06-18/index.html | 1903 +++++ ai/logs/2023-06-21/index.html | 1884 +++++ ai/logs/2023-06-22/index.html | 1894 +++++ ai/logs/2023-07-01/index.html | 1894 +++++ ai/logs/2023-07-23/index.html | 1828 +++++ ai/logs/2023-11-04/index.html | 1766 +++++ ai/logs/2023-11-10/index.html | 1722 +++++ ai/logs/2024-01-28/index.html | 1736 +++++ ai/logs/2024-01-30/index.html | 1955 +++++ ai/logs/2024-01-31/index.html | 1769 +++++ ai/logs/2024-08-16/index.html | 2048 +++++ ai/logs/laytency/index.html | 1754 +++++ ai/match/index.html | 1714 +++++ ai/network/index.html | 1953 +++++ ai/nodes/game_analyzer/index.html | 1754 +++++ ai/nodes/play_switcher/index.html | 1840 +++++ ai/offense/index.html | 1991 +++++ ai/other_team/Sumatra/index.html | 2196 ++++++ ai/other_team/ZJUNLict/index.html | 1710 +++++ ai/other_team/img.png | Bin 0 -> 48467 bytes ai/other_team/rione_racoon_ai_beta/index.html | 1991 +++++ ai/other_team/robodragons/index.html | 1744 +++++ ai/other_team/tigers/index.html | 2528 +++++++ ai/progress/index.html | 2034 +++++ ai/rule/index.html | 2907 +++++++ ai/rvo2_local_planner/index.html | 1826 +++++ ai/setup/index.html | 1826 +++++ ai/simple_ai/index.html | 1772 +++++ ai/skill/index.html | 2275 ++++++ ai/tools/index.html | 1794 +++++ ai/vision/index.html | 2098 ++++++ assets/images/favicon.png | Bin 0 -> 1870 bytes assets/javascripts/bundle.88dd0f4e.min.js | 16 + assets/javascripts/bundle.88dd0f4e.min.js.map | 7 + assets/javascripts/lunr/min/lunr.ar.min.js | 1 + assets/javascripts/lunr/min/lunr.da.min.js | 18 + assets/javascripts/lunr/min/lunr.de.min.js | 18 + assets/javascripts/lunr/min/lunr.du.min.js | 18 + assets/javascripts/lunr/min/lunr.el.min.js | 1 + assets/javascripts/lunr/min/lunr.es.min.js | 18 + assets/javascripts/lunr/min/lunr.fi.min.js | 18 + assets/javascripts/lunr/min/lunr.fr.min.js | 18 + assets/javascripts/lunr/min/lunr.he.min.js | 1 + assets/javascripts/lunr/min/lunr.hi.min.js | 1 + assets/javascripts/lunr/min/lunr.hu.min.js | 18 + assets/javascripts/lunr/min/lunr.hy.min.js | 1 + assets/javascripts/lunr/min/lunr.it.min.js | 18 + assets/javascripts/lunr/min/lunr.ja.min.js | 1 + assets/javascripts/lunr/min/lunr.jp.min.js | 1 + assets/javascripts/lunr/min/lunr.kn.min.js | 1 + assets/javascripts/lunr/min/lunr.ko.min.js | 1 + assets/javascripts/lunr/min/lunr.multi.min.js | 1 + assets/javascripts/lunr/min/lunr.nl.min.js | 18 + assets/javascripts/lunr/min/lunr.no.min.js | 18 + assets/javascripts/lunr/min/lunr.pt.min.js | 18 + assets/javascripts/lunr/min/lunr.ro.min.js | 18 + assets/javascripts/lunr/min/lunr.ru.min.js | 18 + assets/javascripts/lunr/min/lunr.sa.min.js | 1 + .../lunr/min/lunr.stemmer.support.min.js | 1 + assets/javascripts/lunr/min/lunr.sv.min.js | 18 + assets/javascripts/lunr/min/lunr.ta.min.js | 1 + assets/javascripts/lunr/min/lunr.te.min.js | 1 + assets/javascripts/lunr/min/lunr.th.min.js | 1 + assets/javascripts/lunr/min/lunr.tr.min.js | 18 + assets/javascripts/lunr/min/lunr.vi.min.js | 1 + assets/javascripts/lunr/min/lunr.zh.min.js | 1 + assets/javascripts/lunr/tinyseg.js | 206 + assets/javascripts/lunr/wordcut.js | 6708 +++++++++++++++++ .../workers/search.6ce7567c.min.js | 42 + .../workers/search.6ce7567c.min.js.map | 7 + assets/stylesheets/main.6f8fc17f.min.css | 1 + assets/stylesheets/main.6f8fc17f.min.css.map | 1 + assets/stylesheets/palette.06af60db.min.css | 1 + .../stylesheets/palette.06af60db.min.css.map | 1 + index.html | 1619 ++++ search/search_index.json | 1 + sitemap.xml | 3 + sitemap.xml.gz | Bin 0 -> 127 bytes 100 files changed, 104911 insertions(+) create mode 100644 .nojekyll create mode 100644 404.html create mode 100644 ai/attacker/index.html create mode 100644 ai/ball_possession/index.html create mode 100644 ai/coordinates/index.html create mode 100644 ai/diagram/index.html create mode 100644 ai/docker/index.html create mode 100644 ai/grSim/index.html create mode 100644 ai/index.html create mode 100644 ai/logs/2023-04-16/index.html create mode 100644 ai/logs/2023-05-03/index.html create mode 100644 ai/logs/2023-05-06/index.html create mode 100644 ai/logs/2023-05-07/index.html create mode 100644 ai/logs/2023-05-13/index.html create mode 100644 ai/logs/2023-05-14/index.html create mode 100644 ai/logs/2023-05-27/index.html create mode 100644 ai/logs/2023-05-28/index.html create mode 100644 ai/logs/2023-05-30/index.html create mode 100644 ai/logs/2023-06-06/index.html create mode 100644 ai/logs/2023-06-07/index.html create mode 100644 ai/logs/2023-06-12/index.html create mode 100644 ai/logs/2023-06-17/index.html create mode 100644 ai/logs/2023-06-18/index.html create mode 100644 ai/logs/2023-06-21/index.html create mode 100644 ai/logs/2023-06-22/index.html create mode 100644 ai/logs/2023-07-01/index.html create mode 100644 ai/logs/2023-07-23/index.html create mode 100644 ai/logs/2023-11-04/index.html create mode 100644 ai/logs/2023-11-10/index.html create mode 100644 ai/logs/2024-01-28/index.html create mode 100644 ai/logs/2024-01-30/index.html create mode 100644 ai/logs/2024-01-31/index.html create mode 100644 ai/logs/2024-08-16/index.html create mode 100644 ai/logs/laytency/index.html create mode 100644 ai/match/index.html create mode 100644 ai/network/index.html create mode 100644 ai/nodes/game_analyzer/index.html create mode 100644 ai/nodes/play_switcher/index.html create mode 100644 ai/offense/index.html create mode 100644 ai/other_team/Sumatra/index.html create mode 100644 ai/other_team/ZJUNLict/index.html create mode 100644 ai/other_team/img.png create mode 100644 ai/other_team/rione_racoon_ai_beta/index.html create mode 100644 ai/other_team/robodragons/index.html create mode 100644 ai/other_team/tigers/index.html create mode 100644 ai/progress/index.html create mode 100644 ai/rule/index.html create mode 100644 ai/rvo2_local_planner/index.html create mode 100644 ai/setup/index.html create mode 100644 ai/simple_ai/index.html create mode 100644 ai/skill/index.html create mode 100644 ai/tools/index.html create mode 100644 ai/vision/index.html create mode 100644 assets/images/favicon.png create mode 100644 assets/javascripts/bundle.88dd0f4e.min.js create mode 100644 assets/javascripts/bundle.88dd0f4e.min.js.map create mode 100644 assets/javascripts/lunr/min/lunr.ar.min.js create mode 100644 assets/javascripts/lunr/min/lunr.da.min.js create mode 100644 assets/javascripts/lunr/min/lunr.de.min.js create mode 100644 assets/javascripts/lunr/min/lunr.du.min.js create mode 100644 assets/javascripts/lunr/min/lunr.el.min.js create mode 100644 assets/javascripts/lunr/min/lunr.es.min.js create mode 100644 assets/javascripts/lunr/min/lunr.fi.min.js create mode 100644 assets/javascripts/lunr/min/lunr.fr.min.js create mode 100644 assets/javascripts/lunr/min/lunr.he.min.js create mode 100644 assets/javascripts/lunr/min/lunr.hi.min.js create mode 100644 assets/javascripts/lunr/min/lunr.hu.min.js create mode 100644 assets/javascripts/lunr/min/lunr.hy.min.js create mode 100644 assets/javascripts/lunr/min/lunr.it.min.js create mode 100644 assets/javascripts/lunr/min/lunr.ja.min.js create mode 100644 assets/javascripts/lunr/min/lunr.jp.min.js create mode 100644 assets/javascripts/lunr/min/lunr.kn.min.js create mode 100644 assets/javascripts/lunr/min/lunr.ko.min.js create mode 100644 assets/javascripts/lunr/min/lunr.multi.min.js create mode 100644 assets/javascripts/lunr/min/lunr.nl.min.js create mode 100644 assets/javascripts/lunr/min/lunr.no.min.js create mode 100644 assets/javascripts/lunr/min/lunr.pt.min.js create mode 100644 assets/javascripts/lunr/min/lunr.ro.min.js create mode 100644 assets/javascripts/lunr/min/lunr.ru.min.js create mode 100644 assets/javascripts/lunr/min/lunr.sa.min.js create mode 100644 assets/javascripts/lunr/min/lunr.stemmer.support.min.js create mode 100644 assets/javascripts/lunr/min/lunr.sv.min.js create mode 100644 assets/javascripts/lunr/min/lunr.ta.min.js create mode 100644 assets/javascripts/lunr/min/lunr.te.min.js create mode 100644 assets/javascripts/lunr/min/lunr.th.min.js create mode 100644 assets/javascripts/lunr/min/lunr.tr.min.js create mode 100644 assets/javascripts/lunr/min/lunr.vi.min.js create mode 100644 assets/javascripts/lunr/min/lunr.zh.min.js create mode 100644 assets/javascripts/lunr/tinyseg.js create mode 100644 assets/javascripts/lunr/wordcut.js create mode 100644 assets/javascripts/workers/search.6ce7567c.min.js create mode 100644 assets/javascripts/workers/search.6ce7567c.min.js.map create mode 100644 assets/stylesheets/main.6f8fc17f.min.css create mode 100644 assets/stylesheets/main.6f8fc17f.min.css.map create mode 100644 assets/stylesheets/palette.06af60db.min.css create mode 100644 assets/stylesheets/palette.06af60db.min.css.map create mode 100644 index.html create mode 100644 search/search_index.json create mode 100644 sitemap.xml create mode 100644 sitemap.xml.gz diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 0000000..e69de29 diff --git a/404.html b/404.html new file mode 100644 index 0000000..7ebdfa0 --- /dev/null +++ b/404.html @@ -0,0 +1,1557 @@ + + + + + + + + + + + + + + + + + + + ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ +

404 - Not found

+ +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/attacker/index.html b/ai/attacker/index.html new file mode 100644 index 0000000..22253b8 --- /dev/null +++ b/ai/attacker/index.html @@ -0,0 +1,1674 @@ + + + + + + + + + + + + + + + + + + + + + + + Attackerスキル - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

Attackerスキル

+
    +
  • ENTRY_POINT
  • +
  • FORCED_PASS
  • +
  • CUT_THEIR_PASS
  • +
  • STEAL_BALL
  • +
  • REDIRECT_GOAL_KICK
  • +
  • GOAL_KICK
  • +
  • CLEARING_KICK
  • +
  • STANDARD_PASS
  • +
  • LOW_CHANCE_GOAL_KICK
  • +
  • MOVE_BALL_TO_OPPONENT_HALF
  • +
  • RECEIVE_BALL
  • +
+
graph TD
+    ENTRY_POINT(ENTRY_POINT) -.-> FORCED_PASS_C{味方セットプレー?}
+subgraph セットプレー
+    FORCED_PASS_C -- YES --> FORCED_PASS
+end
+
+%%subgraph 敵ボール対処
+%%    STEAL_BALL_C
+%%end
+
+subgraph シュートチャンスは逃さない
+
+end
+
+subgraph 敵ボール対処
+    FORCED_PASS_C -- NO --> CUT_THEIR_PASS_C{動いている敵ボール?}
+    CUT_THEIR_PASS_C -- NO --> STEAL_BALL_C{止まっている敵ボール & ボールに敵が近い?}
+    CUT_THEIR_PASS_C -- YES --> CUT_THEIR_PASS
+end
+
+subgraph シュートチャンスは逃さない
+    STEAL_BALL_C -- NO --> REDIRECT_GOAL_KICK_C{ボールが遠い&ゴールが見えている}
+    REDIRECT_GOAL_KICK_C -- NO --> GOAL_KICK_C{ボールが近い&ゴールが見えている}
+    REDIRECT_GOAL_KICK_C -- YES --> REDIRECT_GOAL_KICK
+    GOAL_KICK_C -- YES --> GOAL_KICK
+end
+
+subgraph 敵ボール対処
+    STEAL_BALL_C -- YES --> STEAL_BALL
+end
+
+subgraph 状況改善
+    GOAL_KICK_C -- NO --> CLEARING_KICK_C{やばい状況?(未実装)}
+    CLEARING_KICK_C -- NO --> STANDARD_PASS_C{パスできそうな相手がいる?}
+    CLEARING_KICK_C -- YES --> CLEARING_KICK
+    STANDARD_PASS_C -- YES --> STANDARD_PASS
+    STANDARD_PASS_C -- NO --> LOW_CHANCE_GOAL_KICK_C{低確率でもシュートできそう(相手コート限定)?}
+    LOW_CHANCE_GOAL_KICK_C -- NO --> MOVE_BALL_TO_OPPONENT_HALF_C{自コートでボールを持っている?}
+    LOW_CHANCE_GOAL_KICK_C -- YES --> LOW_CHANCE_GOAL_KICK
+    MOVE_BALL_TO_OPPONENT_HALF_C -- YES --> MOVE_BALL_TO_OPPONENT_HALF
+    MOVE_BALL_TO_OPPONENT_HALF_C -- NO --> RECEIVE_BALL
+end
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/ball_possession/index.html b/ai/ball_possession/index.html new file mode 100644 index 0000000..3dd6074 --- /dev/null +++ b/ai/ball_possession/index.html @@ -0,0 +1,2009 @@ + + + + + + + + + + + + + + + + + + + + + + + ボールの所持判定アルゴリズム - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

ボールの所持判定アルゴリズム

+

https://github.com/ibis-ssl/crane/pull/303

+

1. イベント検出

+

ボールへの外力発生イベントを検出する。
+イベントは以下の評価値が閾値を超えたときに発生するものとする。

+
評価値 = abs((前フレームの速度ベクトル) - (現在フレームの速度ベクトル)) / (前フレームの速度) + 0.1)
+
+

これによって、ボールの速度ベクトルに変化があった場合にイベントが発生する。
+前フレームの速度で割っているのは、ボールの速度が速い場合には些細な地面の凹凸でボールの動きが変わりやすかったり、単純にVisionのブレが大きいのを吸収することを防ぐ正規化である。
+また、分母に0.1を足しているのは、ボール速度が0に近づくにつれて評価値が不当に大きくなるのを防ぐためである。

+

この評価値を用いて、ロボットによるボールへのキックや衝突、またボールの壁への衝突などのイベントを検出する。

+

2. ボール所持判定

+

ボールの状態を以下の3つに分類する。

+
    +
  • 敵所持
  • +
  • 味方所持
  • +
  • どちらでもない
  • +
+

この状態を計算するためにまず、「敵所持フラグ」「味方所持フラグ」の2つのフラグを計算する

+

2.1. 敵所持フラグ

+

以下の条件を満たすとき、「敵所持フラグ」は真となる。

+
    +
  1. ボールへの外力発生イベントが発生した
  2. +
  3. ボールに一番近い敵ロボットの距離がボールに十分近い(プルリクエストでは0.3mがしきい値)
  4. +
  5. ボールがキッカーに接触した(判定方法は2.1.1参照)
  6. +
+

3は単にロボットがボールにぶつかっただけで所有権が移るのを防ぐための条件である。

+

2.1.1. キッカーへの接触判定

+

ロボット中心から見たボールの方向と、ロボットのキッカーの方向の差が一定角度以下のとき、ボールはキッカーに接触したと判定する。
+プルリクエストでは0.4rad、つまりキッカーの左右に23度ずつの範囲を接触範囲としている。

+

2.2. 味方所持フラグ

+

「味方所持フラグ」は真となる条件は2通りある。

+

2.2.1. 条件1

+

敵所持フラグと同様の条件を味方ロボットが満たしたとき、「味方所持フラグ」は真となる。

+

2.2.2. 条件2

+

以下の条件を満たすとき、「味方所持フラグ」は真となる。

+
    +
  • ロボットのボールセンサが反応した
  • +
+

2.3. フラグとボール状態の遷移

+

遷移テーブルで「-」は遷移無しを表す。

+

2.3.1. 現在状態が「どちらでもない」のときの遷移テーブル

+ + + + + + + + + + + + + + + + + + + + +
敵所持フラグ(真)敵所持フラグ(偽)
味方所持フラグ(真)-味方所持
味方所持フラグ(偽)敵所持-
+

2.3.2. 現在状態が「敵所持」のとき

+ + + + + + + + + + + + + + + + + + + + +
敵所持フラグ(真)敵所持フラグ(偽)
味方所持フラグ(真)どちらでもない味方所持
味方所持フラグ(偽)-どちらでもない
+

2.3.3. 現在状態が「味方所持」のとき

+ + + + + + + + + + + + + + + + + + + + +
敵所持フラグ(真)敵所持フラグ(偽)
味方所持フラグ(真)どちらでもない-
味方所持フラグ(偽)敵所持どちらでもない
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/coordinates/index.html b/ai/coordinates/index.html new file mode 100644 index 0000000..3c6bacc --- /dev/null +++ b/ai/coordinates/index.html @@ -0,0 +1,1698 @@ + + + + + + + + + + + + + + + + + + + + + + + 座標系に関して - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

座標系に関して

+

フィールド座標系

+

Visionから来る座標系 +長辺方向X,短辺方向Y

+

ロボット座標系

+

前X,左Y,上Z

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/diagram/index.html b/ai/diagram/index.html new file mode 100644 index 0000000..2a570b4 --- /dev/null +++ b/ai/diagram/index.html @@ -0,0 +1,1841 @@ + + + + + + + + + + + + + + + + + + + + + + + ノードダイアグラム - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

ノードダイアグラム

+

シミュレータ環境

+
graph TD
+    LP[Local Planner]
+    PS[Play Switcher]
+    GA[Game Analyzer]
+
+    subgraph interface
+        VC[Vision Component]
+        GrC[GrSim Component]
+        GCC[Game Controller Component]
+        Receiver[Robot Receiver]
+    end
+
+    SC[Session Controller]
+    SS[Sim Sender]
+
+    VT[Vision Tracker]
+    WP[World Model Publisher]
+
+    subgraph software
+        GrSim[GrSim]
+        GC[Game Controller]
+    end
+
+    WP -- /world_model -->  SC
+    WP -- /world_model -->  GA
+    GA -- /game_analysis -->  SC
+
+
+    GrC -. UDP .->  GrSim
+
+    GrSim -. UDP .->  VC
+    VC -- /geometry -->  WP
+
+    GrSim -. UDP .->  VC
+
+
+    VC -- /detection -->  VT
+    VT -- /detection_tracked -->  WP
+
+    GC -. UDP .->  GCC
+    GCC -- /referee -->  PS
+    PS -- /play_situation --> SC
+    SC -- /control_targets --> LP
+    LP -- /robot_commands -->  SS
+    SS -- /commands -->  GrC
+
+    Receiver -- /feedback -->  WP
+

実機環境

+
graph TD
+
+    subgraph robot
+        Robots[Robots]
+    end
+
+    subgraph AI Computer
+        LP[Local Planner]
+        PS[Play Switcher]
+
+        VC[Vision Component]
+        GCC[Game Controller Component]
+        Receiver[Robot Receiver]
+
+        SC[Session Controller]
+        SS[Real Sender]
+
+        VT[Vision Tracker]
+        WP[World Model Publisher]
+    end
+
+    subgraph software
+        Vision[SSL Vision]
+        GC[Game Controller]
+    end
+
+    WP -- /world_model -->  SC
+    WP -- /world_model -->  PS
+    WP -- /world_model -->  LP
+
+
+
+    Robots -. UDP .-> Receiver
+    VC -- /geometry -->  WP
+
+    Vision -. UDP .->  VC
+
+
+    VC -- /detection -->  VT
+    VT -- /detection_tracked -->  WP
+
+    GC -. UDP .->  GCC
+    GCC -- /referee -->  PS
+    PS -- /play_situation --> SC
+    PS -- /play_situation --> WP
+    SC -- /control_targets --> LP
+    LP -- /robot_commands -->  SS
+    SS -. UDP .->  Robots
+
+    Receiver -- /feedback -->  WP
+

テスト用のノードダイアグラム

+
graph TD
+    subgraph interface
+        VisionNode[Vision Component]
+        Sender[Real Sender]
+        Receiver[Robot Receiver]
+    end
+
+    VT[Vision Tracker]
+    WP[World Model Publisher]
+    Main[Simple AI]
+    LP[Local Planner]
+
+    subgraph RealWorld
+        Robot[Actual Robot CM4]
+        SSLVision[SSL Vision]
+    end
+
+    SSLVision -. UDP .->  VisionNode
+    VisionNode -- /detection -->  VT
+    VT -- /detection_tracked -->  WP
+    VisionNode -- /geometry -->  WP
+
+    WP -- /world_model -->  Main
+    Main -- /control_targets --> LP
+    LP -- /robot_commands -->  Sender
+
+    Sender -. UDP .->  Robot
+
+    Robot -. UDP .->  Receiver
+    Receiver -- /feedback -->  WP
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/docker/index.html b/ai/docker/index.html new file mode 100644 index 0000000..1c6b23a --- /dev/null +++ b/ai/docker/index.html @@ -0,0 +1,1781 @@ + + + + + + + + + + + + + + + + + + + + + + + docker立ち上げ - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

docker立ち上げ

+

準備

+

docker compose v2のインストール

+
sudo mkdir -p /usr/local/lib/docker/cli-plugins
+sudo curl -SL https://github.com/docker/compose/releases/download/v2.5.0/docker-compose-linux-x86_64 -o /usr/local/lib/docker/cli-plugins/docker-compose
+sudo chmod a+x /usr/local/lib/docker/cli-plugins/docker-compose
+
+

※docker-compose v1系はv1.2.5以降だとエラーが出て起動できないはず

+

起動

+

ツール群

+
cd path/to/crane/docker
+docker compose up
+
+

grSim

+

起動

+

閲覧

+ + + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/grSim/index.html b/ai/grSim/index.html new file mode 100644 index 0000000..69f9e10 --- /dev/null +++ b/ai/grSim/index.html @@ -0,0 +1,1628 @@ + + + + + + + + + + + + + + + + + + + + + + + grSim - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

grSim

+

https://github.com/ibis-ssl/grSim

+

オリジナルから以下の変更を加えた

+
    +
  • ドリブルするときにボールに一定の力がかかるとボールを離すようにした
  • +
  • ペナルティエリアの大きさに関するフィールド(オプショナル)を出力するようにした
      +
    • これは公式Visionでは出力されるが,grSimでは出力されていなかった
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/index.html b/ai/index.html new file mode 100644 index 0000000..e5e6525 --- /dev/null +++ b/ai/index.html @@ -0,0 +1,1620 @@ + + + + + + + + + + + + + + + + + + + + + + + crane Documentation - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

crane Documentation

+

AI components for RoboCup Small Size League.

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-04-16/index.html b/ai/logs/2023-04-16/index.html new file mode 100644 index 0000000..422d708 --- /dev/null +++ b/ai/logs/2023-04-16/index.html @@ -0,0 +1,1768 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-04-16 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-04-16

+

色と位置について

+

色と位置の関係についてはルールには明記されていない.
+しかし,grSimでは,xがマイナスの方が青,プラスの方が黄色となっている.

+

欲しい物とやるべきこと

+

「味方の〇〇 => 座標」を解決したい

+

ある情報は以下の通り

+
    +
  • 味方チームの色情報
  • +
  • 各色チームのランドマーク位置情報
  • +
+

やるべきこと:WorldModelクラスで解決済みの情報を提供するAPIを追加する

+

Defense Area

+

キーパーしか入ってはいけないあのエリア,GoalAreaと表記してしまっていた部分があったが実はDefenseAreaであった.

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_defense_area

+

修正した

+

更新周期について

+

前回調べたら,メインループが300Hzもの超高速で回っていることが判明したので修正していく +原因はおそらく,WorldModelの更新がVisionの一部でも更新されたらPublishされており, +それをトリガーにしてメインループが全て駆動していることが原因だと思われる

+

⇒ WorldModelの更新周期を60Hzに固定しよう!

+

⇒ そもそもWorldModelの更新周期は100Hzだった まぁ早すぎるので60Hzにした

+

local_plannerは/control_targets駆動.
+それを出しているプランナー群はworld_model駆動...

+

あれ,world_model駆動だおかしいな...

+

もしかして,前回測定時にはworld_model_publisherが3つ同時起動していた可能性がある?!?!

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-03/index.html b/ai/logs/2023-05-03/index.html new file mode 100644 index 0000000..0810285 --- /dev/null +++ b/ai/logs/2023-05-03/index.html @@ -0,0 +1,1876 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-03 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-03

+

インプレイ判定のチェック(ルール5.4)

+
    +
  • 以下のコマンド送信後にボールが少なくとも0.05m動く
      +
    • キックオフ
    • +
    • フリーキック
    • +
    • ペナルティーキック
    • +
    +
  • +
  • キックオフから10秒経過
  • +
  • FORCE_STARTコマンド送信後
  • +
+

開発環境整備

+
    +
  • pre-commit導入
  • +
  • include文の呼び分け
  • +
  • boostの警告文抑制
  • +
+

GUIへの現状の表示

+
    +
  • play_switcher.launchへのGUI追加
  • +
+

動作がおかしい?

+

ログ

+
[crane_play_switcher]: last 4, THEIR_KICKOFF_PREPARATION : 2.025060
+[crane_play_switcher]: last 4, THEIR_KICKOFF_PREPARATION : 2.049855
+[crane_play_switcher]: コマンド変化!
+[crane_play_switcher]: THEIR_KICKOFF_START
+[crane_play_switcher]: inplay cmd : 22
+[crane_play_switcher]: CMD : 2
+[crane_play_switcher]: last 2, THEIR_KICKOFF_START : 0.000022
+[crane_play_switcher]: コマンド変化!
+[crane_play_switcher]:
+[crane_play_switcher]: inplay cmd : 0
+[crane_play_switcher]: CMD : 2
+[crane_play_switcher]: last 2,  : 0.000019
+[crane_play_switcher]: last 2,  : 0.024619
+
+

COMMAND_PREPARE_KICKOFF_YELLOWが来て2秒後に NORMAL_STARTが来ている
+その後,

+

参考情報

+
    +
  • COMMAND_NORMAL_START = 2
  • +
  • COMMAND_PREPARE_KICKOFF_YELLOW = 4
  • +
  • inplay/halt = 0
  • +
  • inplay/their_kickoff_start = 22
  • +
+

RAWコマンドの移り変わりとその解釈

+
    +
  1. 2:NORMAL_START,前回のコマンドが残っている?
  2. +
  3. 0:HALT,試合開始
  4. +
  5. 1:STOP,RESUMEを押した
  6. +
  7. 4:PREPARE_KICKOFF_YELLOW,KICKOFFした
  8. +
  9. 2:NORMAL_START, 2秒経って勝手に始まる
  10. +
  11. 2:NORMAL_START, 10秒たってINPLAYに入った
  12. +
+

結局直った気がする

+

⇒色々あって結局デバッグしやすいLOGを整えたら直った気がする +こんな感じ

+
[crane_play_switcher]: ---
+[crane_play_switcher]: RAW_CMD      : COMMAND_STOP
+[crane_play_switcher]: INPLAY_CMD   : STOP
+[crane_play_switcher]: REASON       : RAWコマンド変化:コマンド転送
+[crane_play_switcher]: PREV_CMD_TIME: 12.505770
+[crane_play_switcher]: ---
+[crane_play_switcher]: RAW_CMD      : COMMAND_PREPARE_KICKOFF_YELLOW
+[crane_play_switcher]: INPLAY_CMD   : THEIR_KICKOFF_PREPARATION
+[crane_play_switcher]: REASON       : RAWコマンド変化:コマンド転送
+[crane_play_switcher]: PREV_CMD_TIME: 15.174960
+[crane_play_switcher]: ---
+[crane_play_switcher]: RAW_CMD      : COMMAND_NORMAL_START
+[crane_play_switcher]: INPLAY_CMD   :
+[crane_play_switcher]: REASON       : RAWコマンド変化&NORMAL_START:KICKOFF/PENALTYはPREPARATIONからSTARTに移行
+[crane_play_switcher]: PREV_CMD_TIME: 19.425025
+[crane_play_switcher]: ---
+[crane_play_switcher]: RAW_CMD      : COMMAND_NORMAL_START
+[crane_play_switcher]: INPLAY_CMD   : INPLAY
+[crane_play_switcher]: REASON       : INPLAY判定:キックオフから10秒経過
+[crane_play_switcher]: PREV_CMD_TIME: 10.024822
+
+

コマンド更新通知

+

PlaySituation.msgcommand_updatedを追加した.

+

INPLAYのコマンド更新時にcommand_updatedtrueにする処理を実装した

+

次回

+

PlaySituationを受け取って行う処理の実装

+
    +
  • session_controllerの実装
      +
    • 主に,コマンドを受け取ってセッションを組む部分の実装(ここも泥臭そう...)
    • +
    +
  • +
  • session_controllerで使う上で足りないプランナのリストアップ
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-06/index.html b/ai/logs/2023-05-06/index.html new file mode 100644 index 0000000..578d953 --- /dev/null +++ b/ai/logs/2023-05-06/index.html @@ -0,0 +1,2027 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-06 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-06

+

タスクリスト

+

PlaySituationを受け取って行う処理の実装

+
    +
  • session_controllerの実装
      +
    • 主に,コマンドを受け取ってセッションを組む部分の実装(ここも泥臭そう...)
    • +
    +
  • +
  • session_controllerで使う上で足りないプランナのリストアップ
  • +
+

session_controllerでの実装箇所確認

+

入力

+

topic : /play_situation
+from : play_switcher
+msg type : crane_msgs/PlaySituation.msg

+

msg content :

+
uint32 stage
+uint32 command
+string referee_text
+
+crane_msgs/InPlaySituation inplay_situation
+  bool ball_possession_ours
+  bool ball_possession_theirs
+  uint8 nearest_to_ball_robot_id_ours
+  uint8 nearest_to_ball_robot_id_theirs
+
+geometry_msgs/Point placement_position
+
+crane_msgs/WorldModel world_model
+
+bool command_updated
+
+

役割担当の見直し

+

Before

+
    +
  • play switcher
      +
    • referee コマンドの解釈
    • +
    • ボール保持判定など
    • +
    • world_modelの転送
    • +
    +
  • +
+

After

+
    +
  • play switcher
      +
    • referee コマンドの解釈
    • +
    +
  • +
  • game analyzer
      +
    • ボール保持判定など
    • +
    +
  • +
+

world_modelの受取は必要なものが各自行う

+

ボール保持判定のgame analyzerへの移行

+

msgの更新

+
    +
  • crane_msgs/PlaySituation.msgからボール保持判定の削除
  • +
  • game analyzerから出力するためのボール保持判定のメッセージ検討
  • +
  • crane_msgs/PlaySituation.msgからupdateフラグを削除
  • +
  • crane_msgs/PlaySituation.msgからworld{_modelを削除
  • +
+

すこしGameAnalyzer周りの整備

+
    +
  • game_analyzerの出力メッセージの定義に組み込む
  • +
  • game analyzerのノードドキュメントを追加
  • +
+

session_controllerの実装

+

イベント駆動

+
    +
  • topicの購読
      +
    • /game_analysis
    • +
    • /play_situation
    • +
    +
  • +
+

メンバ変数の命名規則の更新

+

メンバ変数の末尾にアンダースコアを付けるのはやめにした

+

次回のタスク

+
    +
  • session_controllerの実装
      +
    • timer駆動ではなく,イベント駆動にする
    • +
    • 各イベントについての設定を外部ファイルから読み込めるようにする
    • +
    • 代表的なイベントについてプリセットを用意
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-07/index.html b/ai/logs/2023-05-07/index.html new file mode 100644 index 0000000..507b40f --- /dev/null +++ b/ai/logs/2023-05-07/index.html @@ -0,0 +1,1979 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-07 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-07

+

タスクリスト

+
    +
  • session_controllerの実装
      +
    • timer駆動ではなく,イベント駆動にする
    • +
    • 各イベントについての設定を外部ファイルから読み込めるようにする
    • +
    • 代表的なイベントについてプリセットを用意
    • +
    +
  • +
+

session_controllerをイベント駆動にする

+

timer駆動部を削除

+
    +
  • timer駆動部の処理の確認
      +
    • まさかの空だったので削除
    • +
    +
  • +
  • では今までは何駆動で動いてた?
      +
    • コンストラクタで初回だけ呼び出してデバッグしていたようだ
    • +
    +
  • +
  • とりあえずtimerを削除
  • +
+

イベント駆動で動くようにする

+
    +
  • イベントの整理
      +
    • Referee入力( /play_situation
    • +
    • 状況判断による切り替え申請( /game_analysis
    • +
    • Plannerからの失敗報告による切り替え
    • +
    +
  • +
  • 実装の進め方
      +
    • まず,前2つについて実装
        +
      • 切り替えるときに共通で呼び出す関数を作っておく
      • +
      +
    • +
    • Plannerからの失敗報告は,その他の共通インタフェースを整備する良い機会なので別途時間をとって実装
    • +
    +
  • +
  • 切替時に呼び出す共通の関数
      +
    • 結局 request関数を呼び出すのでそれで代用しても良さそう?
        +
      • とりあえずこれで進めて必要あれば後から修正
      • +
      +
    • +
    +
  • +
+

Referee入力駆動の実装

+
    +
  • Referee入力とリアサイン申請のセットを定義する
      +
    • 文字列キーとの相性が悪いのでReferee入力と文字列の対応表が必要
        +
      • play_switcherで作ったやつ
      • +
      • 2箇所で同じものを作るのは微妙なので,PlaySituationのmsg_wrapperを作ってそこで定義する
          +
        • PlaySwitcherもこれを使って書き換え
        • +
        +
      • +
      +
    • +
    • PlaySituationイベントはプレフィックス play_situation/ をつける
    • +
    +
  • +
+

SessionController設定の外部ファイルでの定義

+

とりあえずこんな感じ

+
---
+name: HALT
+description: HALT
+sessions:
+  - name: waiter
+    capacity: 20
+
+

読み込みはyaml-cppを使う +今後の改良ポイントとしては,

+
    +
  • 実行前にエラーを検出するためにyamlのバリデーションをしたい +
  • +
  • 設定ファイルの書き方をyamlからtomlに変更したい
      +
    • yamlはインデントがめんどくさい
    • +
    • tomlはインデントがめんどくさい
    • +
    • どっちもめんどくさい
    • +
    +
  • +
+

InPlaySituation.msgの整理

+
    +
  • reasonを載せるようにした
  • +
  • rawコマンドを載せるようにした
  • +
+

CMakeファイルのクリーンアップ

+
    +
  • インデント幅が2だったり4だったり
      +
    • 2に統一
    • +
    +
  • +
  • C++バージョンが14だったり17だったり
      +
    • 17に統一
    • +
    +
  • +
  • コメントアウトでめっちゃ汚い
      +
    • 不必要なコメントアウトは削除
    • +
    +
  • +
  • 不要に長い空行は削除
  • +
+

Ri-oneの古いAI

+

https://github.com/Rione/ssl-RACOON-AI-beta/tree/competition

+

攻撃時はとりあえずシュートするプログラムシンプルなプログラムでびっくり(INDIRECTを除く)
+ファーストバージョンはこれくらいで良いのかも

+

詳細はこちらに書いた

+

シンプルなバージョンのAIの計画を立ててみる

+
---
+name: SIMPLE_ATTACK
+description: SIMPLE ATTACK
+sessions:
+  - name: "goalie"
+    capacity: 1
+  - name: "shooter"
+    capacity: 1
+  - name: defender
+    capacity: 4
+  - name: waiter
+    capacity: 20
+
+

よく考えれば,シュート出来そうになかったらパスするだけなのでこれがパスは後から組み込みで全然良いなぁ

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-13/index.html b/ai/logs/2023-05-13/index.html new file mode 100644 index 0000000..f657c8b --- /dev/null +++ b/ai/logs/2023-05-13/index.html @@ -0,0 +1,1781 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-13 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-13

+

タスクリスト

+
    +
  • 設定ファイルを読み込めるようにする
      +
    • yaml-cppを使って読み込む
    • +
    • 読み込んだ設定を使ってsession_controllerを初期化する
    • +
    +
  • +
+

設定ファイルを読み込めるようにする

+
    +
  • yaml-cppを使った読み込み
      +
    • YAMLファイル以外は弾く
    • +
    • サブディレクトリを検知して再帰読み込み
    • +
    +
  • +
  • 読み込んだ内容を内部構造体に落とし込み
  • +
  • ハードコードされた設定をファイルに落とし込み
  • +
+

シチュエーションと設定ファイルの対応付け

+

これもまた設定ファイルを作った方が良い

+
    +
  • ひとまず,シチュエーションごとのセッション設定ファイルは play_situation/ 以下にまとめる
      +
    • 設定ファイルの移動
    • +
    • 設定ファイルの読み込みのディレクトリの変更
    • +
    +
  • +
  • 対応ファイルの作成:event_config.yaml
  • +
  • PlaySituationイベントのコールバックを作成
  • +
+

まだまだ設定とそれに伴う実装が足りてないけど,枠組みとしてはいい感じ

+
[play_switcher_node-10] [INFO] [1683989045.103454434] [crane_play_switcher]: ---
+[play_switcher_node-10] [INFO] [1683989045.103487097] [crane_play_switcher]: RAW_CMD      : 1 (COMMAND_STOP)
+[play_switcher_node-10] [INFO] [1683989045.103501253] [crane_play_switcher]: INPLAY_CMD   : 1 (STOP)
+[play_switcher_node-10] [INFO] [1683989045.103513245] [crane_play_switcher]: REASON       : RAWコマンド変化:コマンド転送
+[play_switcher_node-10] [INFO] [1683989045.103526384] [crane_play_switcher]: PREV_CMD_TIME: 5.388861
+[crane_session_controller_node-5] [ERROR] [1683989045.103671870] [session_controller]: イベント「STOP」に対応するセッションの設定が見つかりませんでした
+[play_switcher_node-10] [INFO] [1683989048.678499392] [crane_play_switcher]: ---
+[play_switcher_node-10] [INFO] [1683989048.678535396] [crane_play_switcher]: RAW_CMD      : 0 (COMMAND_HALT)
+[play_switcher_node-10] [INFO] [1683989048.678548954] [crane_play_switcher]: INPLAY_CMD   : 0 (HALT)
+[play_switcher_node-10] [INFO] [1683989048.678557748] [crane_play_switcher]: REASON       : RAWコマンド変化:コマンド転送
+[play_switcher_node-10] [INFO] [1683989048.678567988] [crane_play_switcher]: PREV_CMD_TIME: 8.963902
+[crane_session_controller_node-5] [INFO] [1683989048.678675503] [session_controller]: イベント「HALT」に対応するセッション「HALT」の設定に従ってロボットを割り当てます
+[crane_session_controller_node-5] [INFO] [1683989048.678703304] [session_controller]: request : HALT
+[crane_session_controller_node-5] [INFO] [1683989048.678712896] [session_controller]: selectable_robot_ids : 0 1 2 3 4 5 6 7 8 9 10
+
+

次回タスク

+
    +
  • ある程度セッションの設定を作ってしまいたい
  • +
  • ”attacker”のプランナ実装に着手する
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-14/index.html b/ai/logs/2023-05-14/index.html new file mode 100644 index 0000000..81d247c --- /dev/null +++ b/ai/logs/2023-05-14/index.html @@ -0,0 +1,1823 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-14 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-05-14

+

雑務

+ +

AttackerPlanner

+

役割:ボールへ突撃する役

+

実装方針

+

インダイレクトのときは別のプランナで...と分けたくなるのを我慢していったんここへ押し込める

+

シチュエーションと動作

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
シチュエーション動作
STOPボールの一番近くで待機
INPLAYボールに突進してシュートする,ダメそうなら近くの味方にパスを出す
KICKOFF_PREPARATIONボールの一番近くで待機
KICKOFF_START近くの味方にパスする
INDIRECT_FREE近くの味方にパスを出す
DIRECT_FREE近くの味方にパスを出す
PENALTY_PREPARATIONボールの一番近くで待機
PENALTY_STARTシュートする
+

割当ポリシー

+

ボールに一番近いロボットを割り当て

+

代表的な割当ポリシーは名前付きでいくつかプリセット実装をしても置いて良いかも

+
    +
  • ボールに近い
  • +
  • 敵 / 味方ゴールに近い
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-27/index.html b/ai/logs/2023-05-27/index.html new file mode 100644 index 0000000..aed62eb --- /dev/null +++ b/ai/logs/2023-05-27/index.html @@ -0,0 +1,1844 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-27 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-27

+

タスク

+

動作確認@ハンス宅のための作業

+
    +
  • Visionの設定
  • +
  • ミニコートのためのコード変更など
  • +
+

Visionの設定

+

docs/vision.mdを参照

+

Rootsのと雷さんの助けもあり,無事に動作確認できた +https://twitter.com/ShotaAk/status/1662225994735050752

+

ミニコートのためのコード変更など

+

実験のためのlaunchファイルを作成

+

通信まわりをチェック

+

JoyStickで遊べるようにする

+

ミニコートを考慮していない部分をあぶり出してひたすら修正していく

+

課題

+
    +
  • local_plannerが動かない
  • +
  • コントローラのコンフィグ
      +
    • 動かすと必ずドリブラがONになる
    • +
    +
  • +
  • ロボット座標系への変換がSimSenderにいる
      +
    • local_plannerに移動したい
    • +
    +
  • +
  • ゴールキーパーだけのモードを作る
  • +
  • ペナルティキックをできるようにする
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-28/index.html b/ai/logs/2023-05-28/index.html new file mode 100644 index 0000000..1db6d94 --- /dev/null +++ b/ai/logs/2023-05-28/index.html @@ -0,0 +1,1832 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-28 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-05-28

+

タスク

+
    +
  • local_plannerが動かない
  • +
  • コントローラのコンフィグ
      +
    • 動かすと必ずドリブラがONになる + ~~- ロボット座標系への変換がSimSenderにいる
    • +
    • local_plannerに移動したい~~
    • +
    +
  • +
  • ゴールキーパーだけのモードを作る
  • +
  • ペナルティキックをできるようにする
  • +
+

ロボット座標系への変換をlocal_plannerに移動

+

移動先の決定

+

そもそも,local_plannerには色々な処理が混在しているので関数で分割する +やっぱり座標変換はsenderでやるべきかも +SimとRealの共通処理を設けるべきな気がしてきた

+

Senderに共通のロボット座標系への変換処理を実装

+
    +
  • SenderBaseクラスを設ける
  • +
  • 共通処理を移動
  • +
  • SimSender,RealSenderでこれを継承する
  • +
  • ロボット座標系の変換を実装
  • +
+

練習合宿で最低限やりたいこと

+

リソース:Crow 1台,Orin 1台

+
    +
  • ディフェンスの動作確認
      +
    • 2台ディフェンス
    • +
    • 1台ディフェンス1台キーパー
    • +
    +
  • +
  • PK戦
      +
    • 1台キッカー1台キーパー
    • +
    +
  • +
+

次回のタスク

+
    +
  • LocalPlanner通すとロボット動かない問題のデバッグ
  • +
  • crane_teleopのリファクタ
      +
    • コントローラのコンフィグを変える
    • +
    • 操作方法のドキュメント
    • +
    • DualShockで操作できるようにする
    • +
    +
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-05-30/index.html b/ai/logs/2023-05-30/index.html new file mode 100644 index 0000000..07288ea --- /dev/null +++ b/ai/logs/2023-05-30/index.html @@ -0,0 +1,1939 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-05-30 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+ +
+
+ + + +
+
+ + + + + + + +

2023-05-30

+

タスク

+
    +
  • LocalPlanner通すとロボット動かない問題のデバッグ
  • +
+

初期イベント

+
    +
  • 初期イベントをパラメータから読み込めるように
  • +
  • セッションと同名のイベントを作成するようにした
      +
    • 新しいイベントに対するセッションを作る時,同名のセッションを作るだけで良くなった
    • +
    +
  • +
+

GUIの表示

+

RobotCommandの更新に追従できていなかったので修正

+

LocalPlannerのデバッグ

+
    +
  • RVOなしモードの実装
  • +
  • なんか動いた
  • +
+

チームカラー対応確認

+

sim_sender

+

ちゃんと色どおりのものが動いた

+

そもそも

+

そもそも,チームカラー情報はレフェリーからくる +なので,/refereeをWorldModelPublisherが受信してチーム情報を更新する +レフェリー関連を起動しないときもあるので初期値はros paramで設定できるようにする

+

WorldModelPublisher

+
    +
  • 初期チームカラーを設定できるように
  • +
  • /refereeを読んでチームカラーを設定できるように
  • +
+

Planners

+

Goalie

+

見事にその場で青チームのゴール位置をハードコートしていたので修正

+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Defender
        +
      • DefenseArea周りのWorldModelWrapperのAPIを整備
      • +
      +
    • +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • 動かした時の動きがおかしい問題の修正
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-06/index.html b/ai/logs/2023-06-06/index.html new file mode 100644 index 0000000..deee121 --- /dev/null +++ b/ai/logs/2023-06-06/index.html @@ -0,0 +1,1811 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-06 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-06-06

+

タスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Defender
        +
      • DefenseArea周りのWorldModelWrapperのAPIを整備
      • +
      +
    • +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • 動かした時の動きがおかしい問題の修正
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+

Plannerが青チーム前提になっているのを修正(Defender)

+
    +
  • world_modelが色々おかしい
      +
    • penalty_area_width, penalty_area_heightにゼロが入っている
        +
      • そもそもVisionのパケットにゼロが入ってる
      • +
      • 取り敢えずworld_model_publisherで固定値を出すように
      • +
      +
    • +
    • サイズと位置がややこしい
        +
      • 取り違えていたりしたのでWorldModel.msgのフィールド名うぃ変更したりした
      • +
      +
    • +
    +
  • +
  • WorldModelWrapperにゴールの位置を取得できるAPIを追加
  • +
  • ↑を使って書き直し
  • +
  • 目標位置はうまく行ってそう
      +
    • コマンド送信先は変わらない
    • +
    +
  • +
+

コマンド送信先がおかしいのを直す

+
    +
  • おかしい部分の特定
      +
    • GameControllerからのデータはしっかり受け取れている
    • +
    • WorldModelにもしっかり流れている
    • +
    • RobotCommandsに設定できていない!!!
        +
      • した
      • +
      +
    • +
    +
  • +
+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • 動かした時の動きがおかしい問題の修正
      +
    • local_planner?
    • +
    +
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-07/index.html b/ai/logs/2023-06-07/index.html new file mode 100644 index 0000000..aa446a5 --- /dev/null +++ b/ai/logs/2023-06-07/index.html @@ -0,0 +1,1766 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-07 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-06-07

+

タスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • 動かした時の動きがおかしい問題の修正
      +
    • local_planner?
    • +
    +
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+

動かした時の動きがおかしい問題の修正

+
    +
  • 原因追求
      +
    • current_poseが代入されていなかった
    • +
    • 座標変換が間違っていた
    • +
    • スピードの符号が逆だった
    • +
    • 周波数の設定が間違っていた
    • +
    +
  • +
+

色々修正したらようやく直った

+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-12/index.html b/ai/logs/2023-06-12/index.html new file mode 100644 index 0000000..aba7bdc --- /dev/null +++ b/ai/logs/2023-06-12/index.html @@ -0,0 +1,1755 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-12 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-06-12

+

タスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • 新しいlaunchファイル
      +
    • ゴールキーパーだけ
    • +
    • ディフェンダーだけ
    • +
    +
  • +
  • ペナルティキックの実装
  • +
+

crane_teleopのリファクタ

+

real_senderの表示するIDを変える

+

ros2 paramで対応 +parameterのコールバック初めて書いた

+

次のタスク

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-17/index.html b/ai/logs/2023-06-17/index.html new file mode 100644 index 0000000..bd997cf --- /dev/null +++ b/ai/logs/2023-06-17/index.html @@ -0,0 +1,1896 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-17 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-06-17

+

タスク

+
    +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • ペナルティキックの実装
  • +
+

crane_teleopのリファクタ

+
    +
  • DualSense前提に書き換え
  • +
  • 操作方法を変えた
  • +
  • ros2 paramでIDなどを変更できるように
  • +
+

キックオフの実現

+

PREPARATION

+
    +
  • kickoffプランナで布陣につく
  • +
+

START

+
    +
  • ATTACKERプランナを割り当て
  • +
+

Attackerプランナ実装

+

ボールの後ろに回り込んでひたすら相手ゴールにシュートするプランナを作った. +後ろへの回り込み判定はボールを中心に相手ゴールとの方向に対して後ろにいるかどうかを内積で判定した.

+

ControlTargetライブラリの充実

+

- 円弧に沿っての移動

+

RobotReceiverについて

+

蹴ったかどうかの判定が非常に難しいので,ロボットからのフィードバックがほしい. +オカダがUDPに既に投げてくれているので,あとはそれを拾ってROS 2に持ってくるだけではある

+ +

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • session_controllerでロボット割当するときの初期ロボット列をworld_modelから入れる
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-18/index.html b/ai/logs/2023-06-18/index.html new file mode 100644 index 0000000..10e29f5 --- /dev/null +++ b/ai/logs/2023-06-18/index.html @@ -0,0 +1,1903 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-17 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-06-17

+

タスク

+
    +
  • Attackerの改良
      +
    • ボールが外に出ると中心に戻るように
    • +
    • デフォルトで敵ゴールを向くように
    • +
    +
  • +
+

crane_teleopのリファクタ

+
    +
  • DualSense前提に書き換え
  • +
  • 操作方法を変えた
  • +
  • ros2 paramでIDなどを変更できるように
  • +
+

キックオフの実現

+

PREPARATION

+
    +
  • kickoffプランナで布陣につく
  • +
+

START

+
    +
  • ATTACKERプランナを割り当て
  • +
+

Attackerプランナ実装

+

ボールの後ろに回り込んでひたすら相手ゴールにシュートするプランナを作った. +後ろへの回り込み判定はボールを中心に相手ゴールとの方向に対して後ろにいるかどうかを内積で判定した.

+

ControlTargetライブラリの充実

+

- 円弧に沿っての移動

+

RobotReceiverについて

+

蹴ったかどうかの判定が非常に難しいので,ロボットからのフィードバックがほしい. +オカダがUDPに既に投げてくれているので,あとはそれを拾ってROS 2に持ってくるだけではある

+ +

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    • Attacker
    • +
    +
  • +
  • session_controllerでロボット割当するときの初期ロボット列をworld_modelから入れる
  • +
+
    +
  • crane_teleopのリファクタ
  • +
  • コントローラのコンフィグを変える
  • +
  • 操作方法のドキュメント
  • +
  • DualShockで操作できるようにする
  • +
  • ペナルティキックの実装
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-21/index.html b/ai/logs/2023-06-21/index.html new file mode 100644 index 0000000..f6fb56d --- /dev/null +++ b/ai/logs/2023-06-21/index.html @@ -0,0 +1,1884 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-06-21 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-06-21

+

タスク

+
    +
  • プランナの実行失敗からの再割当てまでのフローを固める
  • +
  • フォーメーションの実装
  • +
+

プランナの実行失敗からの再割当てまでのフローを固める

+

ブレスト的なの

+
    +
  • 実行失敗?
      +
    • world_modelなどを入力にして判定関数を定義する
    • +
    • 役割の責任失敗とは違う
        +
      • ゴールキーパーがゴールされたからといってここでいう失敗に当たらない
      • +
      +
    • +
    • 再割当てしたほうが良い状況になったとき
        +
      • 毎フレーム他のロボットのスコアも確認する必要あり
          +
        • game_analyzerとかでやったほうが良くない?
        • +
        +
      • +
      +
    • +
    +
  • +
  • 失敗したときに考えられる次の行動
      +
    • 同じプランナに別のロボットを再割当てして該当プランナだけやり直し
    • +
    • 全ロボット再割当て(場合によってはセッション切り替え)
    • +
    +
  • +
  • 実行失敗の実際
      +
    • Receiverが
    • +
    +
  • +
+

構造変更?

+
    +
  • ロボット割り当てやその時のスコア計算はgame_analyzerでやるべき
      +
    • robot_assignerの名前の方がいい?
    • +
    • session_controllerでやればええやん
    • +
    +
  • +
  • スコアは毎フレーム計算して適宜再割当てする
      +
    • 再割当て後のコストが高いSessionについてはHysteresisコストを設ける
    • +
    +
  • +
+

構造変更|実践編

+
    +
  • SessionModuleないを改造していく
      +
    • 計算はModule内でやって計算結果だけを渡す感じ
    • +
    • スコア計算関数は計算するだけの処理にする
    • +
    • 割当関数は実際に通信して割り当てる
    • +
    • いくつかの関数は実装をpluginから持ってきたい
        +
      • スコア計算関数
      • +
      +
    • +
    +
  • +
+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    +
  • +
  • session_controllerでロボット割当するときの初期ロボット列をworld_modelから入れる
  • +
  • ペナルティキックの実装
  • +
+

RobotReceiver

+

蹴ったかどうかの判定が非常に難しいので,ロボットからのフィードバックがほしい. +オカダがUDPに既に投げてくれているので,あとはそれを拾ってROS 2に持ってくるだけではある

+ + + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-06-22/index.html b/ai/logs/2023-06-22/index.html new file mode 100644 index 0000000..5a6d1d7 --- /dev/null +++ b/ai/logs/2023-06-22/index.html @@ -0,0 +1,1894 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-07-01 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-07-01

+

タスク

+
    +
  • プランナの実行失敗からの再割当てまでのフローを固める
  • +
  • フォーメーションの実装
  • +
+

プランナの実行失敗からの再割当てまでのフローを固める

+

構造変更?

+
    +
  • ロボット割り当てやその時のスコア計算はsession_controllerでやる
      +
    • 関数定義はそれぞれの
    • +
    +
  • +
  • スコアは毎フレーム計算して適宜再割当てする
      +
    • 再割当て後のコストが高いSessionについてはHysteresisコストを設ける
    • +
    +
  • +
+

構造変更|実践編

+
    +
  • SessionModule内を改造していく
      +
    • 計算はModule内でやって計算結果だけを渡す感じ
    • +
    • スコア計算関数は計算するだけの処理にする
    • +
    • 割当関数は実際に通信して割り当てる
    • +
    • いくつかの関数は実装をpluginから持ってきたい
        +
      • スコア計算関数
      • +
      +
    • +
    +
  • +
+

構造変更|プラグイン化

+
    +
  • SessionModuleの処理をPlannerBaseに移植
  • +
  • 各プランナのプラグイン登録
  • +
  • plugins.xml更新
  • +
  • 各プランナクラスの書き換え
      +
    • プラグインにする場合,コンストラクタの引数はなしにする必要がある
    • +
    +
  • +
+

構造変更|受け入れ体制整備

+

骨組み

+
update
+  judgeSessionResult
+  updateRoleScore
+  judgeReassignSession
+
+
    +
  • 有効なセッションとそうじゃないセッションをどうやって管理する?
  • +
+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    +
  • +
  • session_controllerでロボット割当するときの初期ロボット列をworld_modelから入れる
  • +
  • ペナルティキックの実装
  • +
+

RobotReceiver

+

蹴ったかどうかの判定が非常に難しいので,ロボットからのフィードバックがほしい. +オカダがUDPに既に投げてくれているので,あとはそれを拾ってROS 2に持ってくるだけではある

+ + + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-07-01/index.html b/ai/logs/2023-07-01/index.html new file mode 100644 index 0000000..a6ca0f5 --- /dev/null +++ b/ai/logs/2023-07-01/index.html @@ -0,0 +1,1894 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-07-01 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2023-07-01

+

タスク

+
    +
  • プランナの実行失敗からの再割当てまでのフローを固める
  • +
  • フォーメーションの実装
  • +
+

プランナの実行失敗からの再割当てまでのフローを固める

+

構造変更?

+
    +
  • ロボット割り当てやその時のスコア計算はsession_controllerでやる
      +
    • 関数定義はそれぞれの
    • +
    +
  • +
  • スコアは毎フレーム計算して適宜再割当てする
      +
    • 再割当て後のコストが高いSessionについてはHysteresisコストを設ける
    • +
    +
  • +
+

構造変更|実践編

+
    +
  • SessionModule内を改造していく
      +
    • 計算はModule内でやって計算結果だけを渡す感じ
    • +
    • スコア計算関数は計算するだけの処理にする
    • +
    • 割当関数は実際に通信して割り当てる
    • +
    • いくつかの関数は実装をpluginから持ってきたい
        +
      • スコア計算関数
      • +
      +
    • +
    +
  • +
+

構造変更|プラグイン化

+
    +
  • SessionModuleの処理をPlannerBaseに移植
  • +
  • 各プランナのプラグイン登録
  • +
  • plugins.xml更新
  • +
  • 各プランナクラスの書き換え
      +
    • プラグインにする場合,コンストラクタの引数はなしにする必要がある
    • +
    +
  • +
+

構造変更|受け入れ体制整備

+

骨組み

+
update
+  judgeSessionResult
+  updateRoleScore
+  judgeReassignSession
+
+
    +
  • 有効なセッションとそうじゃないセッションをどうやって管理する?
  • +
+

次のタスク

+
    +
  • Plannerが青チーム前提になっているのを修正
      +
    • Receiver
    • +
    • Waiter
    • +
    +
  • +
  • session_controllerでロボット割当するときの初期ロボット列をworld_modelから入れる
  • +
  • ペナルティキックの実装
  • +
+

RobotReceiver

+

蹴ったかどうかの判定が非常に難しいので,ロボットからのフィードバックがほしい. +オカダがUDPに既に投げてくれているので,あとはそれを拾ってROS 2に持ってくるだけではある

+ + + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-07-23/index.html b/ai/logs/2023-07-23/index.html new file mode 100644 index 0000000..b243f85 --- /dev/null +++ b/ai/logs/2023-07-23/index.html @@ -0,0 +1,1828 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-07-23 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-07-23

+

タスク

+

通信内容の見直し

+

通信内容の見直し

+

内容

+
    +
  • Visionのxy
  • +
  • 遅れ時間
  • +
  • ボール相対座標,速度
  • +
+

msg更新

+
    +
  • Visionのxy
      +
    • current_poseをそのまま
    • +
    +
  • +
  • 遅れ時間
      +
    • latency_msを追加
    • +
    +
  • +
  • ボール相対座標,速度
      +
    • ball_relative_pose, velocity[x,y]を追加
    • +
    +
  • +
+

関連クラス更新

+
    +
  • wrapperクラス
  • +
  • TypeAdapter
  • +
+

sender更新

+

次のタスク

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-11-04/index.html b/ai/logs/2023-11-04/index.html new file mode 100644 index 0000000..361a346 --- /dev/null +++ b/ai/logs/2023-11-04/index.html @@ -0,0 +1,1766 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-11-04 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-11-04

+

Skillを中心としたシステムを組む

+

Skillを中心としたシステムへの移行目処が立った,というかxx_plannerの中身は一部Skillで置き換えている. +これは,Skillの試用を目的とする暫定的な処置であるが,良さげであったので,このままSkillを中心としたシステムを組むことにした.

+

現状と今後

+
    +
  • 基礎の動き
      +
    • 現状:xx_plannerに直接実装
        +
      • 使い回しは基本的にできず,コードが冗長になる
      • +
      +
    • +
    • 今後:Skill
    • +
    +
  • +
  • 役割
      +
    • 現状:xx_plannerが該当
        +
      • 同一インスタンスが使い回されるため,しっかりリセットが必要
      • +
      +
    • +
    • 今後:Role
        +
      • ROS上のインスタンスは持たせず,上位層ノードで管理する
      • +
      • RoleはJITで生成・破棄する
      • +
      +
    • +
    +
  • +
  • 状況による割り振り
      +
    • 現状:session_controllerが該当
    • +
    • 今後:ベースは同じでOK
        +
      • Roleに対応した実装にする
      • +
      • 後から別の割当アルゴリズムの別実装も作れるように考えておく
      • +
      +
    • +
    +
  • +
+

今日の作業

+

-[x] plugin_baseをRole用に書き換え -[x] サービス通信をなくす -[x] session_planner側作業 -[x] SessionModuleをplugin_baseに置き換え

+
    +
  • 置き換え後の動作確認
  • +
  • local plannerの各種パラメータの外だし
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2023-11-10/index.html b/ai/logs/2023-11-10/index.html new file mode 100644 index 0000000..060a61f --- /dev/null +++ b/ai/logs/2023-11-10/index.html @@ -0,0 +1,1722 @@ + + + + + + + + + + + + + + + + + + + + + + + 2023-11-10 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

2023-11-10

+

動かす会に向けたメモ

+

用意するデモの種類

+
    +
  • ボールプレイスメント(1台)
      +
    • ゆっくりやりたいので最高速度を個別に設定できるようにする
    • +
    • 急回転をどうにかする
    • +
    • 設定ファイルの用意
    • +
    +
  • +
  • パスシュート(2台(vs 1台))
      +
    • 敵ロボットとして同時にうまく動かせる?
    • +
    • 設定ファイルの用意
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2024-01-28/index.html b/ai/logs/2024-01-28/index.html new file mode 100644 index 0000000..5574a1f --- /dev/null +++ b/ai/logs/2024-01-28/index.html @@ -0,0 +1,1736 @@ + + + + + + + + + + + + + + + + + + + + + + + 2024-01-28 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+ +
+
+ + + +
+
+ + + + + + + +

2024-01-28

+

直近でやることの整理

+
    +
  • SimpleAIのアップグレード
      +
    • Skillも使えるようにしたい
        +
      • Skillのパラチュンができるようにしたい
      • +
      +
    • +
    • RobotCommandWrapperネイティブにできたらいいな
    • +
    • +

      機体の状態を取得できるようにしたい

      +
    • +
    +
  • +
  • 回避のデバッグ
      +
    • 取り敢えずまずこれをやる
    • +
    +
  • +
+

回避の問題点

+

生成した回避点が、既に通り過ぎたものを含んでいる。 +こうなると、途中で戻ったり立ち往生したりしてしまう。

+

現在は、ロボット側から回避点をたどってゴールに到着できるか判断しているが、 +ゴール側からたどるとうまく行くかも?

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2024-01-30/index.html b/ai/logs/2024-01-30/index.html new file mode 100644 index 0000000..ea1ca7f --- /dev/null +++ b/ai/logs/2024-01-30/index.html @@ -0,0 +1,1955 @@ + + + + + + + + + + + + + + + + + + + + + + + 2024-01-30 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2024-01-30

+

直近でやることの整理

+
    +
  • SimpleAIのアップグレード
      +
    • Skillも使えるようにしたい
        +
      • Skillのパラチュンができるようにしたい
      • +
      +
    • +
    • RobotCommandWrapperネイティブにできたらいいな
    • +
    • 機体の状態を取得できるようにしたい
    • +
    +
  • +
+

SimpleAIの拡張を見据えて

+

すべての機能をSkill化するのはあり。

+

Skillとは?

+
    +
  • 入力
      +
    • ID
    • +
    • WorldModel
    • +
    • パラメータ
    • +
    • 引き継ぎ情報(コンテキスト)
    • +
    +
  • +
  • 出力
      +
    • ロボットコマンド
    • +
    • Skillの終了判定
    • +
    • 引き継ぎ情報(コンテキスト)
    • +
    +
  • +
+

ひとまずRobotCommandをSkill化する

+

マクロを駆使して、RobotCommandをSkill化した。 +しかし、パラメータをうまく渡せないので、うまく噛み合わない。

+

そういえば

+

Skillのパラメータを共通インタフェースを渡せるようにすると型の問題などが面倒なので、 +各個スキルクラスでパラメータのセッターを独自定義することにしたんだった。 +このときは、Skillクラスをプログラム上で呼び出すことしか考えていなかったので、 +特に共通インタフェースを無理して作るインセンティブもなく、むしろ各々のスキルクラスで便利なように独自定義できることが強みだと思っていた。

+

とはいえ

+

SimpleAIなど、共通インタフェースを使ってパラメータを渡したい場合が出てきた今、そうも言っていられない。 +ということで、共通インタフェースを作ることにした。

+

パラメータ融通用共通インタフェース設計

+

現状

+

SimpleAIでは、vectorを使っている。 +既に存在するスキルを見ても、doubleで表現できるパラメータしかない。

+

今後を見据えて

+
    +
  • boolは、0.0と1.0で表現することで、doubleで表現できるが、double以外も扱うなら対応したい。
  • +
  • intは、doubleで表現できるが、double以外も扱うなら対応したい。
  • +
  • stringは、対応していると何かと便利そう。
  • +
+

ということで

+

将来を見据えると、複数の型を扱える共通インタフェースがほしい

+

std::vector>かな +そして、名前付きパラメータにしたいので、std::unordered_map>かな

+

実際の実装

+
    +
  • 共通インタフェースrun()の引数に、std::unordered_map>を追加
      +
    • std::optionalにして、nulloptのときは動作させないことで、引数なしのrun()も可能にする - これで後方互換性が保たれる。- これで、関数型っぽくも振る舞えるはず。
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2024-01-31/index.html b/ai/logs/2024-01-31/index.html new file mode 100644 index 0000000..f42f186 --- /dev/null +++ b/ai/logs/2024-01-31/index.html @@ -0,0 +1,1769 @@ + + + + + + + + + + + + + + + + + + + + + + + 2024-01-31 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+ +
+ + + +
+
+ + + + + + + +

2024-01-31

+

直近でやることの整理

+
    +
  • SimpleAIのアップグレード
      +
    • Skillも使えるようにしたい
        +
      • Skillのパラチュンができるようにしたい
      • +
      +
    • +
    • RobotCommandWrapperネイティブにできたらいいな
    • +
    • 機体の状態を取得できるようにしたい
    • +
    +
  • +
+

SimpleAIの拡張を見据えて

+

昨日は、インタフェースを設計して、SkillBaseクラスに対して設置するところまで行った。 +今日は、既存のSkillの対応と、SimpleAIでの対応を行う。

+

既存のSkillの対応

+
    +
  • GetBallContact
  • +
  • Idle
  • +
  • MoveToGeometry
  • +
  • MoveWithBall
  • +
  • TurnAroundPoint
  • +
  • RobotCommandベースのスキル
  • +
+

SimpleAIのパラメータスキル対応

+

既存のUIだと無理なので、パラメータテーブルを使ってたくさんのパラメータを設定できるようにする。

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/2024-08-16/index.html b/ai/logs/2024-08-16/index.html new file mode 100644 index 0000000..f21cce7 --- /dev/null +++ b/ai/logs/2024-08-16/index.html @@ -0,0 +1,2048 @@ + + + + + + + + + + + + + + + + + + + + + + + 2024-08-16 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

2024-08-16

+

パッケージ巡礼

+

3rdparty/closest_point_vendor

+

Boost.Geometryの拡張機能。最近傍点を求めるために使っている。 +ビルドにはインターネット接続が必要。

+

3rdparty/matplotlib_cpp_17_vendor

+

matplotlibのC++ラッパー。無印のmatplotlib-cppとは別物で高機能。 +今は使っていないが、軽いデバッグに便利

+

consai_ros2/robocup_ssl_msgs

+

https://github.com/SSL-Roots/consai_ros2

+

SSL公式のprotoメッセージをROS2で使えるようにしたもの。 +GameEvent関連は移植されていない。

+

consai_ros2/robocup_ssl_comm

+
    +
  • game_controller
  • +
  • grsim
      +
    • ロボットコマンドを送信したり、ボールやロボット位置の配置指示
    • +
    +
  • +
  • robot_status
      +
    • grsim上のロボット状態を受信
    • +
    +
  • +
  • vision
  • +
+

consai2_examples/consai_vision_tracker

+

EKFでVisionデータをフィルタリングして、使いやすいようにするパッケージ

+

crane_world_model_publisher

+

SSL関連のデータをすべて集めたトピックである/world_modelを配信するパッケージ +これをすることで、他のノードは/world_modelだけを購読すればよくなる

+

utility/crane_msg_wrappers

+

world_model_wrapperは、world_modelの情報を使って、よく使う幾何学計算をまとめたもの

+

utility/crane_basics

+

基礎的な幾何学計算などを集めたパッケージ

+
    +
  • ボールの接触時間を計算するための時系列計算を行う機能(ball_contact)
  • +
  • WorldModelWrapperのための構造体(robot_info, ball_info)
  • +
  • Boost.GeometryでEigenのPosition型をネイティブに扱えるようにするための設定(eigen_adapter) +
  • +
  • PID制御器(pid_controller)
  • +
  • 基礎的な幾何学計算(geometry_operations)
      +
    • 角度の正規化など角度の不連続性を無視して計算できる関数など
    • +
    +
  • +
  • 区間計算(interval)
      +
    • Boostに類似機能があるのでそっちに移行してもいいかも
    • +
    +
  • +
  • 最適割当問題のソルバー(position_assignments)
      +
    • ハンガリアン法を実装している
    • +
    • 複数ロボットの割当を行うときに使っている(defense_plannerなど)
    • +
    • 最適割当ではないような場面も見るのでバグがあるかも? +
    • +
    +
  • +
  • 移動時間計算(travel_time)
      +
    • 台形加速を考慮したロボットの移動時間を計算する
    • +
    • テストも実装してある(test/test.cpp)
    • +
    +
  • +
  • ボールモデルの計算 +
  • +
+

スキル作るにあたってやること

+

スキルクラスを実装する

+
    +
  • update関数
  • +
  • print関数
  • +
+

SimpleAIに登録する

+
    +
  • crane_robot_skills/skills.hppに作ったスキルのヘッダファイルを追加
  • +
  • crane_simple_ai/crane_commander.cppでスキルを登録
      +
    • setUpSkillDictionary関数で登録
    • +
    +
  • +
+

対応するPlannerを作る

+

実際の動きに組み込むためには、Plannerを作る必要がある。

+
    +
  • crane_planner_pluginsに作る
  • +
  • Skill単体のPlannerはskill_planner.hppに実装することが多い
  • +
+

Plannerの登録

+

crane_planner_plugins/planners.hppに文字列とPlannerのペアを登録する

+

SimpleAIとgrSim

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/logs/laytency/index.html b/ai/logs/laytency/index.html new file mode 100644 index 0000000..c893897 --- /dev/null +++ b/ai/logs/laytency/index.html @@ -0,0 +1,1754 @@ + + + + + + + + + + + + + + + + + + + + + + + 遅延のボトルネック解析 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

遅延のボトルネック解析

+

https://github.com/ibis-ssl/crane/issues/130

+

Timer/Callback駆動

+

まずは現状分析

+
graph TD
+    LP["`Callback
+    **Local Planner**
+    Immediately`"]
+    PS[Play Switcher]
+    GA[Game Analyzer]
+
+    subgraph interface
+        VC["`100Hz
+        **Vision Component**
+        Immediately`"]
+        GrC[GrSim Component]
+        GCC[Game Controller Component]
+        Receiver[Robot Receiver]
+    end
+
+    SC["`WorldModel Callback
+    **Session Controller**
+    Immediately`"]
+    SS[Sim Sender]
+
+    VT["`100Hz
+        **Vision Tracker**
+        Immediately`"]
+    WP["`Callback
+        **World Model Publisher**
+        30Hz`"]
+
+    subgraph software
+        GrSim[GrSim]
+        GC[Game Controller]
+    end
+
+    WP -- /world_model -->  SC
+    WP -- /world_model -->  GA
+    GA -- /game_analysis -->  SC
+
+
+    GrC -. UDP .->  GrSim
+
+    GrSim -. UDP .->  GrC
+    GrC -- /geometry -->  WP
+
+    GrSim -. UDP .->  VC
+
+
+    VC -- /detection -->  VT
+    VT -- /detection_tracked -->  WP
+
+    GC -. UDP .->  GCC
+    GCC -- /referee -->  PS
+    PS -- /play_situation --> SC
+    SC -- /control_targets --> LP
+    LP -- /robot_commands -->  SS
+    SS -- /commands -->  GrC
+
+    Receiver -- /feedback -->  WP
+
    +
  • 下流はほとんどコールバック駆動なので良し
  • +
  • Vision Component
      +
    • 非同期で受け取ってUDPコールバックにしても良いかも
    • +
    • これは絶対に受け取った時刻を一緒に情報として持っておくべき
    • +
    +
  • +
  • Vision Tracker
      +
    • フィルター処理はタイマー駆動必要かも
    • +
    • これ以降の処理はコールバックで一気通貫したい
        +
      • 柔軟性的にはフィルター処理コールバック周期と出力周期は変えたいかも
      • +
      • いい感じ補間処理入れたら出力周期変えてもpublish時の最新の情報を出せる?
      • +
      +
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/match/index.html b/ai/match/index.html new file mode 100644 index 0000000..9e4b4cd --- /dev/null +++ b/ai/match/index.html @@ -0,0 +1,1714 @@ + + + + + + + + + + + + + + + + + + + + + + + 試合チェックリスト - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

試合チェックリスト

+

Vision

+
    +
  • アドレスあってる?
  • +
  • 違うインタフェースでマルチキャストジョインしたりしない?
      +
    • netstat -g でマルチキャストグループとインタフェースの対応を確認
    • +
    • WiFiをONにしていると、WiFiのインタフェースでマルチキャストグループに参加してしまうことが多い
    • +
    +
  • +
+

GameController

+
    +
  • アドレスあってる?
      +
    • 調整用にポート変えたやつを使っていないか?
    • +
    +
  • +
  • マルチキャストジョインしている?
      +
    • netstat -g でマルチキャストグループとインタフェースの対応を確認
    • +
    • WiFiをONにしていると、WiFiのインタフェースでマルチキャストグループに参加してしまうことが多い
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/network/index.html b/ai/network/index.html new file mode 100644 index 0000000..7ee6b74 --- /dev/null +++ b/ai/network/index.html @@ -0,0 +1,1953 @@ + + + + + + + + + + + + + + + + + + + + + + + ネットワーク設定 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

ネットワーク設定

+

ROS関連

+

https://autowarefoundation.github.io/autoware-documentation/pr-347/installation/additional-settings-for-developers/#network-settings-for-ros-2

+

ローカルホストでマルチキャスト

+
sudo ip link set multicast on lo
+
+

マルチキャストアドレスとデバイスの対応の確認

+
netstat -g
+
+

マルチキャストアドレスへのルートの追加

+
sudo ip route add <address> dev <device>
+
+

インターネット接続とロボット接続の共存

+

ロボットのアドレスに対して静的ルーティングを設定する

+

/etc/netplan/01-network-manager-all.yaml

+
network:
+  version: 2
+  renderer: networkd
+  ethernets:
+    enp3s0:
+      addresses:
+        - 192.168.1.2/24
+      gateway4: 192.168.1.1
+      routes:
+        - to: 192.168.2.0/24
+          via: 192.168.1.1
+        - to: 192.168.3.0/24
+          via: 192.168.1.1
+
+
graph TD
+    subgraph official
+        GameController[Game Controller]
+        AutoRef[Auto Ref]
+        Vision[SSL Vision]
+    end
+
+    OfficialHub[大会用スイッチングハブ]
+
+    subgraph AIPC
+        OfficialInterface[大会サーバー用Interface]
+        ibisInterface[ロボット用Interface]
+        crane[crane]
+        sender[real_sender]
+        receiver[robot_receiver]
+    end
+
+    SwitchingHub[スイッチングハブ]
+    Router[ルーター]
+
+    Robots[ロボット]
+    PC[開発PC]
+
+    Net[インターネット]
+
+    GameController -- UDP Multicast --> OfficialHub
+    AutoRef -- UDP Multicast --> OfficialHub
+    Vision -- UDP Multicast --> OfficialHub
+
+    OfficialHub -- UDP Multicast --> SwitchingHub
+    SwitchingHub -- UDP Multicast --> OfficialInterface
+    OfficialInterface -- UDP Multicast --> crane
+    crane -- ROS --> sender
+    sender -- UPD to 192.168.20.1xx --> ibisInterface
+
+    ibisInterface -- UPD to 192.168.20.1xx --> SwitchingHub
+    SwitchingHub -- UPD to 192.168.20.1xx --> Router
+    Router -- AICommand --> Robots
+    Robots -- RobotFeedback --> Router
+    Router -- UPD to 192.168.20.1xx --> SwitchingHub
+    SwitchingHub -- RobotFeedback UPD Multicast --> ibisInterface
+    ibisInterface -- RobotFeedback UPD Multicast --> receiver
+    receiver -- ROS  --> crane
+
+

アドレス・ポートなど

+

公式ツールなど

+
    +
  • Vision
      +
    • アドレス:224.5.23.2
    • +
    • ポート
        +
      • 10006
          +
        • 本番で使われることが多い
        • +
        • ssl-visionのデフォルトポート
        • +
        +
      • +
      • 10020
          +
        • シミュレーションなどで使われることが多い
        • +
        • grSimのデフォルトポート
        • +
        +
      • +
      +
    • +
    +
  • +
  • Game Controller
      +
    • アドレス:224.5.23.1
    • +
    • ポート
        +
      • 10003
          +
        • デフォルト
        • +
        • 本番で使われる
        • +
        +
      • +
      • 11111
          +
        • ibisがポート被り防止に使うことが多い
        • +
        +
      • +
      +
    • +
    +
  • +
  • Tracker +
  • +
+

ibis

+
    +
  • ロボットのCM4
      +
    • アドレス:192.168.20.100+機体番号
    • +
    • コマンド用ポート:12345
    • +
    +
  • +
  • ロボットからのフィードバック
      +
    • アドレス:224.5.20.100
    • +
    • ポート:50100+機体番号
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/nodes/game_analyzer/index.html b/ai/nodes/game_analyzer/index.html new file mode 100644 index 0000000..3ea3366 --- /dev/null +++ b/ai/nodes/game_analyzer/index.html @@ -0,0 +1,1754 @@ + + + + + + + + + + + + + + + + + + + + + + + Game Analyzer - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

Game Analyzer

+

Refereeなどから直接示されない,ゲームの状況を解析するノード.
+ゲームの状況を解析し,ゲームの状況を示すメッセージを毎フレームpublishする.
+重い処理を含む解析は数フレームに1度のみ行い,それ以外のフレームでは前回の結果をpublishする.

+

ボール所持判定

+

両チームのボールに最も近いロボットを抽出し,ボールとの距離がしきい値以下ならばボール所持と判定する.

+

TODO

+

味方チームの戦略から,パス中などもボール所持判定にできるようにする. +そのためには,パスの成否判定の実装が必要である

+

パスの成否判定

+

未実装

+

タスクリスト

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/nodes/play_switcher/index.html b/ai/nodes/play_switcher/index.html new file mode 100644 index 0000000..816df30 --- /dev/null +++ b/ai/nodes/play_switcher/index.html @@ -0,0 +1,1840 @@ + + + + + + + + + + + + + + + + + + + + + + + Play Switcher - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

Play Switcher

+

refereeからの信号を受け取って,解析した情報を流すノード
+refereeからは毎フレーム情報が流れてくるが,PlaySwitcherからの出力時は情報に更新があった時のみ

+

敵味方イベントの解釈

+

Blue / Yellowのイベントを解釈して,敵味方のイベントに変換する

+

NORMAL_STARTの解釈

+

前のイベントに応じてNORMAL_STARTの解釈を行う

+

参考:5.3.1 Normal Start | Official Rule

+

NORMAL_STARTの解釈

+ + + + + + + + + + + + + + + + + +
前のイベントNORMAL_STARTの解釈
KICKOFF_PREPARATIONKICKOFF_START
PENALTY_PREPARATIONPENALTY_START
+

INPLAYの判定

+

INPLAYのイベントはRefereeからは送信されてこないのでPlaySwitcherで判定する

+

参考:5.4. Ball In And Out Of Play | Official Rule

+

INPLAYの判定

+
    +
  • FORCE_START発動時
  • +
  • 以下のイベント後,ボールが少なくとも0.05m動いた時
      +
    • KICKOFF_START
    • +
    • PENALTY_START
    • +
    • DIRECT_FREE
    • +
    • INDIRECT_FREE
    • +
    +
  • +
  • フリーキックで以下の秒数が経過したとき
      +
    • 5秒(Division A)
    • +
    • 10秒(Division B)
    • +
    +
  • +
+

タスクリスト

+
    +
  • STOPの解釈
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/offense/index.html b/ai/offense/index.html new file mode 100644 index 0000000..4bfc724 --- /dev/null +++ b/ai/offense/index.html @@ -0,0 +1,1991 @@ + + + + + + + + + + + + + + + + + + + + + + + オフェンス - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

オフェンス

+

アクション

+

ボールの所有権をもつロボットが、ボールを操作して行う行動をアクションとする。 +ボールは1つなので、複数台のロボットが同時にボールの所有権を持つことはなく、一連のオフェンスはアクションの列として表現される。

+

アクションの本体は、ロボットがボールを操作する瞬間の集合である。 +例えば、パスやシュートならば、ボールを蹴る瞬間であり、ドリブルならばドリブルしている間である。

+

ただし、これだけだとオフェンス中にどのアクションにも属さない状態が生じるので、前のアクションの本体が終了したとき、次のアクション +が始まるものとする。(これらの時間範囲はアクションに含まれるだけであって、アクションの本体であるわけではない) +例えば、パスの場合ではボールが前のロボットからキックされて自分の方に転がり始めた瞬間からパスアクションが始まり、次のロボットへボールを蹴り出した瞬間にパスアクションが終了する。 +ここで、キックに失敗したときは上位レイヤーで認知され、アクション列が作り直される。

+

また、アクションではアクション本体以外のステージも持つ。

+
    +
  • PRE_STAGE: アクション本体が始まる前に行うステージ
  • +
  • ON_STAGE: アクション本体が行われるステージ
  • +
  • POST_STAGE: アクション本体が終わった後に行うステージ
  • +
+

パスアクション

+

PRE_STAGE

+

パスを受ける地点から離れすぎない範囲で、マークを振り切るための動きを行う。

+

暫定実装

+

パスを受け取る地点へ移動する

+

ON_STAGE

+

パスを受け取る地点へ移動し、ボールを受け取って次のターゲットへキックする

+

POST_STAGE

+

特になし

+

シュートアクション

+

正直このアクションは必要かどうかは微妙 +パスアクションの特殊化として扱うべきかもしれない

+

PRE_STAGE

+

ON_STAGE

+

POST_STAGE

+

ドリブルアクション

+

PRE_STAGE

+

ON_STAGE

+

POST_STAGE

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/other_team/Sumatra/index.html b/ai/other_team/Sumatra/index.html new file mode 100644 index 0000000..74d9cc9 --- /dev/null +++ b/ai/other_team/Sumatra/index.html @@ -0,0 +1,2196 @@ + + + + + + + + + + + + + + + + + + + + + + + TIGERs Sumatra - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

TIGERs Sumatra

+

Attackerの状態機械

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/master/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/pandora/roles/offense/attacker/AttackerRole.java

+
stateDiagram-v2
+    Protect --> ApproachBallLine: ballMoves
+    Protect --> Kick : switchToKick
+
+    Dribble --> DribbleKick: switchToDribbleKick
+    Dribble --> Kick: switchToKick
+    Dribble --> Protect: FAILURE
+
+    ApproachBallLine --> Receive: SUCCESS
+    ApproachBallLine --> approachAndStopBall: FAILURE
+    ApproachBallLine --> approachAndStopBall: closeToBall
+
+    approachAndStopBall --> Protect: SUCCESS
+    approachAndStopBall --> Protect: FAILURE
+
+    Kick --> Receive: SUCCESS
+    Kick --> Protect: FAILURE / INVALID
+    Kick --> FreeKick: waitForKick
+    Kick --> FreeKick: useSingleTouch
+
+    FreeKick --> ApproachBallLine: SUCCESS
+    FreeKick --> Protect: FAILURE
+
+    Receive --> Protect: SUCCESS
+    Receive --> Protect: FAILURE
+    Receive --> Redirect: SwitchToRedirect
+
+    Redirect --> ApproachBallLine: SUCCESS
+    Redirect --> Protect: FAILURE
+    Redirect --> Receive: switchToReceive
+
+    DribbleKick --> Protect: dribblingKickIsBlocked
+    DribbleKick --> Protect: FAILUR
+

Keeperの状態機械

+
stateDiagram-v2
+    Stop --> Defend: !stoped
+
+    PreparePenalty --> Defend: !isPreparePenalty
+
+    MoveToPenaltyArea --> Defend: SUCCESS
+    MoveToPenaltyArea --> Defend: isKeeperWellInsidePenaltyArea
+    MoveToPenaltyArea --> Stop: isStopped
+    MoveToPenaltyArea --> PreparePenalty: isPreparePenalty
+
+    Defend --> Pass: ballCanBePassedOutOfPenaltyArea
+    Defend --> Rambo: canGoOut
+    Defend --> GetBallContact: isBallBetweenGoalyAndGoal
+    Defend --> MoveToPenaltyArea: isOutsidePenaltyArea
+    Defend --> Stop: isStopped
+    Defend --> PreparePenalty: isPreparePenalty
+    Defend --> Intercept: canInterceptSafely
+
+    Pass --> Defend: isBallMoving
+    Pass --> MoveInFrontOfBall: ballPlacementRequired
+    Pass --> Stop: isStopped
+    Pass --> PreparePenalty: isPreparePenalty
+
+    Intercept --> Defend: hasInterceptionFailed
+    Intercept --> Pass: ballPlacementRequired
+    Intercept --> Stop: isStopped
+    Intercept --> PreparePenalty: isPreparePenalty
+
+    Rambo --> Defend: isBallInPenaltyArea(0) || isGoalKick()
+    Rambo --> Stop: isStopped
+    Rambo --> PreparePenalty: isPreparePenalty
+
+    MoveInFrontOfBall --> Defend: isBallMoving
+    MoveInFrontOfBall --> Defend: ballPlaced
+    MoveInFrontOfBall --> GetBallContact: SUCCESS
+    MoveInFrontOfBall --> Stop: isStopped
+    MoveInFrontOfBall --> PreparePenalty: isPreparePenalty
+
+    GetBallContact --> MoveWithBall: SUCCESS
+    GetBallContact --> MoveInFrontOfBall: FAILURE
+    GetBallContact --> Stop: isStopped
+    GetBallContact --> PreparePenalty: isPreparePenalty
+
+    MoveWithBall --> Defend: SUCCESS
+    MoveWithBall --> MoveInFrontOfBall: FAILURE
+    MoveWithBall --> Stop: isStopped
+    MoveWithBall --> PreparePenalty: isPreparePenalty
+
+

BallPlacementの状態機械

+
stateDiagram-v2
+    Receive --> Prepare: SUCCESS
+    Receive --> StopBall: FAILURE
+    Prepare --> DropBall: ballIsPlaced
+    Prepare --> GetBallContact: success
+    Prepare --> Receive: ballMoving
+    Prepare --> Pass: ballNeedsToBePassed
+    Prepare --> GetBallContact: skipPrepare
+    StopBall --> Prepare: SUCCESS
+    StopBall --> Prepare: FAILURE
+    GetBallContact --> MoveWithBall: SUCCESS
+    GetBallContact --> DropBall: FAILURE
+    MoveWithBall --> DropBall: SUCCESS
+    MoveWithBall --> Receive: FAILURE
+    Pass --> ClearBall: FAILURE
+    Pass --> ClearBall: SUCCESS
+    Pass --> ClearBall: pass mode is NONE
+    DropBall --> ClearBall: FAILURE
+    DropBall --> ClearBall: SUCCESS
+

AOffensiveActionMoves

+

TDP2018に解説されている。

+
    +
  • ForcedPass
  • +
  • DirectKick
      +
    • 敵ゴールへの直接シュート
    • +
    +
  • +
  • ClearingKick
      +
    • 味方の危ない状況からのクリア
    • +
    +
  • +
  • StandardPass
  • +
  • LowChanceKick
      +
    • 敵ゴールへの直接シュート
    • +
    +
  • +
  • GoToOtherHalf
      +
    • 自陣でボールをキープしているが適切なパスターゲットがない場合
    • +
    +
  • +
  • KickInsBlue
      +
    • 敵ロボットがいない場所にボールを蹴る
    • +
    +
  • +
  • RedirectGoalShot
  • +
  • RedirectPass
  • +
  • Receive
  • +
+

ベースクラス(2018年時点でのもの)

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/e858a04f90d5f9d0c0d9df0e4fd913e66d386e6c/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/offense/action/moves/AOffensiveActionMove.java

+

※viability: 実行可能性

+
    +
  • isActionViable
      +
    • そのアクションが実行可能かどうかを返す
    • +
    • 返り値
        +
      • TRUE
      • +
      • PARTIAL
      • +
      • FALSE
      • +
      +
    • +
    +
  • +
  • activateAction
  • +
  • calViabilityScore
  • +
+

ベースクラス(2021年時点でのもの)

+

calcAction

+

返り値:OffensiveAction

+
    +
  • move:EOffensiveActionMove(PROTECT_MOVE)
      +
    • FORCED_PASS
    • +
    • REDIRECT_GOAL_KICK
    • +
    • GOAL_KICK
    • +
    • CLEARING_KICK
    • +
    • STANDARD_PASS
    • +
    • LOW_CHANCE_GOAL_KICK
    • +
    • MOVE_BALL_TO_OPPONENT_HALF
    • +
    • KICK_INS_BLAUE
    • +
    • RECEIVE_BALL
    • +
    • PROTECT_MOVE
    • +
    +
  • +
  • action:EOffensiveAction(PASS)
      +
    • PASS
    • +
    • KICK_INS_BLAUE
    • +
    • CLEARING_KICK
    • +
    • GOAL_SHOT
    • +
    • PROTECT
    • +
    • REDIRECT
    • +
    • RECEIVE
    • +
    +
  • +
  • viability:OffensiveActionViability
      +
    • TRUE
    • +
    • PARTIAL
    • +
    • FALSE
    • +
    +
  • +
+

ベースクラス(2023年時点でのもの)

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/master/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/offense/action/EOffensiveActionType.java

+
    +
  • type:OffensiveActionType
      +
    • PASS
    • +
    • REDIRECT_KICK
    • +
    • DRIBBLE_KICK
    • +
    • PROTECT
    • +
    • CHOP_TRICK
        +
      • Hackentrick: かかとで後方にけり、相手をかわす技
      • +
      +
    • +
    • RECEIVE
    • +
    +
  • +
+

https://github.com/TIGERs-Mannheim/Sumatra/blob/master/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/offense/action/moves/EOffensiveActionMove.java

+
    +
  • OffensiveActionMoveの種類
      +
    • REDIRECT_GOAL_KICK
    • +
    • CHOP_TRICK
    • +
    • FINISHER
    • +
    • GOAL_KICK
    • +
    • FORCED_PASS
    • +
    • CLEARING_KICK
    • +
    • STANDARD_PASS
    • +
    • LOW_CHANCE_GOAL_KICK
    • +
    • MOVE_BALL_TO_OPPONENT_HALF
    • +
    • RECEIVE_BALL
    • +
    • PROTECT_MOVE
    • +
    +
  • +
+

build関数はOffensiveActionTypeのラインナップをサポート

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/master/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/offense/action/RatedOffensiveAction.java

+
    +
  • RatedOffensiveAction
      +
    • buildProtect
    • +
    • buildChopTrick
    • +
    • buildPass
    • +
    • buildKick
    • +
    • buildRedirectKick
    • +
    • buildDribbleKick
    • +
    • buildReceive
    • +
    +
  • +
+

ForcedPass

+

フリーキックのような、必ずパスしなければいけない状況で使われる。

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/master/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/offense/action/moves/ForcedPassActionMove.java

+

FinisherActionMove

+

ドリブルしながらゴールを狙うアクション

+

ClearingKickActionMove

+

ボールをゴール中心から離れる方向にチップキックする

+
    +
  • TRUE
      +
    • ボールがゴールに近い&壁デフェンスが間に合ってない&ボールが相手に近い&自コートにボールがある&ボールが0.3m/s以下
    • +
    +
  • +
  • FALSE
      +
    • それ以外
    • +
    +
  • +
+

ProtectActionMove

+

PassGenerator

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/5e99731884440591f633e65b01118f291f460d4e/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/pass/PassGenerator.java

+

KIckOriginという概念

+

https://github.com/TIGERs-Mannheim/Sumatra/blob/259774013fe5fef0c5eeff46db5f04c3f7d61f56/modules/moduli-ai/src/main/java/edu/tigers/sumatra/ai/metis/pass/KickOrigin.java

+

次にボールがキックされる点。

+

ボールが動いていないときは、ボールの場所であり、動いているときは以下のいずれかが発生したときである。

+
    +
  • (相手ボールのとき)ボールをカット
  • +
  • ボールをキャッチ
  • +
  • ボールをリダイレクト
  • +
+

impact time: ボールがKickOriginに到着するまでの時間

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/other_team/ZJUNLict/index.html b/ai/other_team/ZJUNLict/index.html new file mode 100644 index 0000000..533656c --- /dev/null +++ b/ai/other_team/ZJUNLict/index.html @@ -0,0 +1,1710 @@ + + + + + + + + + + + + + + + + + + + + + + + ZJUNLict - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + Skip to content + + +
+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

ZJUNLict

+

Dynamic Passing Points Searching(DPPS)

+

v1ではボールを中心に円筒座標系でサンプリングしていたが,v2ではロボットを中心にサンプリングしている

+

TDP注目ポイント

+
    +
  • 2019
      +
    • 新ドリブラ
    • +
    • Implementation of Intercept Ball Skill
    • +
    • Implementation of Marking Skill
    • +
    +
  • +
  • 2014
      +
    • FSM-based Role Matching
    • +
    +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/other_team/img.png b/ai/other_team/img.png new file mode 100644 index 0000000000000000000000000000000000000000..366617b9a8d52347eed58e8b489caf89af13b88b GIT binary patch literal 48467 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
UAI#>%ki3-ik2rpAw{z=y5Xasl8T4N&ZKR)2$ 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 Vzr1!32#|j2ckd~e-^M-re*ngZ27v$o literal 0 HcmV?d00001 diff --git a/ai/other_team/rione_racoon_ai_beta/index.html b/ai/other_team/rione_racoon_ai_beta/index.html new file mode 100644 index 0000000..57c97f0 --- /dev/null +++ b/ai/other_team/rione_racoon_ai_beta/index.html @@ -0,0 +1,1991 @@ + + + + + + + + + + + + + + + + + + + + + + + RACOON AI beta of Ri-one - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +

+
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

RACOON AI beta of Ri-one

+

https://github.com/Rione/ssl-RACOON-AI-beta

+

Role分け

+

https://github.com/Rione/ssl-RACOON-AI-beta/tree/competition/racoon_ai/strategy/role

+

Offense

+

決定戦略

+

https://github.com/Rione/ssl-RACOON-AI-beta/blob/competition/racoon_ai/strategy/role/role.py#L138

+
    +
  1. 有効でキーパーでもDefenseでもないロボットを抽出
  2. +
  3. 相手ゴールに近い順に規定台数を選択.それ以外は削除
  4. +
  5. なにかの角度に並べ替え?
  6. +
+

SubRole

+

https://github.com/Rione/ssl-RACOON-AI-beta/blob/competition/racoon_ai/strategy/role/subrole.py#L53

+
    +
  • our_attacker
      +
    • ボールに一番近いロボットが1台選ばれる
    • +
    +
  • +
  • receiver
      +
    • our_attackerに近いロボットが1台選ばれる
    • +
    +
  • +
+

※enemy_attackerは敵チームのボールに一番近いロボット

+

Scheme

+

defense

+
    +
  • デフォルトポジションがある
  • +
  • ボールがゴールに向かっていたら阻止する?
  • +
+

goal_keeper

+

offense

+
    +
  • indirectならreceiver(一番近い味方)にパス
  • +
  • そうでなければゴールにシュート
  • +
+
    +
  • 関数名
      +
    • block_their_attacker
    • +
    • default_position
    • +
    • penalty_kick
    • +
    • stop_attacker
    • +
    • stop_offense
    • +
    • pass_to_receiver
    • +
    • shoot_to_goal
    • +
    • direct_their
        +
      • ボールに一番近い味方がblock_their_attacker
      • +
      +
    • +
    +
  • +
+

out_of_play

+

Defense

+

その他

+
    +
  • キーパー
  • +
+

コマンド周り

+
    +
  • avoid_penalty_area
  • +
  • avoid_ball
  • +
  • avoid_enemy
  • +
  • speed_limiter
  • +
  • pid(目標位置, ロボット)
  • +
  • to_front_ball
  • +
  • ball_around
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/other_team/robodragons/index.html b/ai/other_team/robodragons/index.html new file mode 100644 index 0000000..ed93da9 --- /dev/null +++ b/ai/other_team/robodragons/index.html @@ -0,0 +1,1744 @@ + + + + + + + + + + + + + + + + + + + + + + + Robodragons - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

Robodragons

+ +

TDP2019

+

到着時間の予測

+
    +
  • 従来は目的地への直線を使って時間予測をしていた
      +
    • 障害物があると正確でなくなる
    • +
    +
  • +
  • 中間点を一つ使って予測する
      +
    • 中間点では停止する前提で予測する
    • +
    • 予測精度が51%向上した
    • +
    +
  • +
+

五芒星パス

+
    +
  • イベントでの魅力的なデモが必要
  • +
  • 5台のロボットが円を描きながら移動する
  • +
  • それらのロボットが五芒星を描くようなパスをする
  • +
  • Rootsによる実装 +
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/other_team/tigers/index.html b/ai/other_team/tigers/index.html new file mode 100644 index 0000000..51db305 --- /dev/null +++ b/ai/other_team/tigers/index.html @@ -0,0 +1,2528 @@ + + + + + + + + + + + + + + + + + + + + + Tigers - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

Tigers

+ +

TDP2023

+

https://tdp.roboteamtwente.nl/tdps/144?ref=year

+

軌道生成

+

ゴールキーパーのような、ゴール地点で速度ゼロにならなくても場所と時間さえあっていればOKというケースに対応して有効守備範囲を広げている。

+

TDP2022

+

https://tdp.roboteamtwente.nl/tdps/241?ref=year

+

ハードウェアアップデート

+
    +
  • ドリブラのダンパ
  • +
  • 自分のマーカーを読み取るシステム
  • +
+

ビルトインなシミュレータについて

+
    +
  • grSimは低レイヤー制御も含むので、AIの検証用に理想的な動作をするシミュレータを作った(2013~)
  • +
  • 基本的に2Dだが、チップキックだけは3D対応
  • +
  • デバッグが簡単
      +
    • ステップ実行可能
    • +
    • 状況を巻き戻して再実行も可能
    • +
    +
  • +
  • シミュレータの比較
      +
    • grSim
    • +
    • ER-Forceのもの
    • +
    +
  • +
  • シミュレータを使ったテストも簡単に作れる
  • +
+

攻撃用ドリブル

+
    +
  • 2021年大会の優勝で大きな役割を果たした
  • +
  • ペナルティエリアの平行線上でサンプリングして評価
  • +
  • サンプリングする点はドリブル最大距離まで
  • +
  • ゴール可能な最近傍点を選んでそこまでドリブルする
  • +
+

視覚化

+ +

データの記録

+
    +
  • 1試合のログは数GB
      +
    • 圧縮したら数百MB
    • +
    +
  • +
  • ログ再生機でコピーしたらロボット・フィールド状態がjson形式でクリップボードに
  • +
  • シミュレータ上でペーストすると状態が再現される
  • +
+

テスト

+

シミュレータを組み込んでのテスト

+
    +
  • パスカット
  • +
  • リダイレクトシュート
  • +
+

ルールのテスト

+
    +
  • 対戦相手なしでFORCE_START:10s以内にゴールできるか
  • +
  • KICKOFF: 11s以内にロボットが動き出すか?
  • +
  • PENALTY: ロボットが動き出し、30s以内にゴールしてHALTになるか外してBallPlacementになっているか?
  • +
  • FORCESTART: ボールが蹴られて動き出すか?
  • +
  • Pass: ロボットが時間内に目的地に到着するか?
  • +
  • STOP: STOP中にロボットが最大速度を超えて移動していないか?禁止エリアに入らないか?
  • +
  • BallPlacement: 時間以内に成功するか?
  • +
  • KICKOFF: 相手無しで11s以内にゴールできるか?
  • +
  • INDIRECT FREE: 6s以内にロボットが移動するか?
  • +
+

TDP2020

+

https://tdp.roboteamtwente.nl/tdps/89?ref=year

+

ハードウェア v2020(Gen5)

+
    +
  • 前のバージョンはv2019
  • +
  • メカ中心のアップデートで回路はほぼ同じ
  • +
+

車輪のアップデート

+

img.png

+
    +
  • v2019の車輪
      +
    • 3Dプリント PLA製
    • +
    • 直径33mm
        +
      • かなり小さいが、車輪が90°間隔で配置できる
      • +
      +
    • +
    • チームフィールドでは問題なかった
    • +
    • シドニー大会で壊れまくった
        +
      • 毛深いカーペットと塗装されたラインが原因
      • +
      • ホイールカバーが壊れてサブホイールが定期的に欠落する
          +
        • サブホイールが欠落するとホイールベースがカーペットに接触して摩擦で溶け始める
            +
          • PLAは60℃で溶ける
          • +
          +
        • +
        +
      • +
      +
    • +
    +
  • +
  • v2020の車輪
      +
    • アルミ製ホイール
    • +
    • 直径62mm
        +
      • 安価に製造するように変更した結果、大きくなった
      • +
      • v2019をアルミで削り出すと高価
      • +
      • 90°間隔で配置できない
      • +
      +
    • +
    • ダイレクトドライブで駆動
    • +
    • サブホイール
        +
      • サブホイールは20個
      • +
      • Xリング
          +
        • 設置点が2箇所でスムーズな動きにつながる
        • +
        • 摩擦もOリングより優れる
        • +
        • v2016の車輪にも使っていたが、ベアリング部分も摩耗でガタがでて結果的に振動が大きくなった
            +
          • 今回はしっかりベアリングを使って対策した
          • +
          +
        • +
        +
      • +
      +
    • +
    +
  • +
+

キッカーのアップデート

+
    +
  • ダンパはTPUの3Dプリンタ製
  • +
  • プランジャはスチールとアルミの組み合わせ
  • +
+

ドリブラのアップデート

+
    +
  • IRセンサアレイ搭載
  • +
  • モーターの変更
      +
    • v2019: 高速で小型なモーターを減速
        +
      • すごいうるさかった
      • +
      +
    • +
    • v2020: 減速なしで55W定格17000rpmのモーターを短期過負荷モードで使用
    • +
    +
  • +
+

カバー

+
    +
  • PETG製 1.2mm厚
      +
    • 色々な素材を試したが、PETGが最もよかった
    • +
    • 適度に柔軟性を持ち、衝撃吸収できる
    • +
    +
  • +
  • 接着剤を塗布している
      +
    • カバーが破損しても接着剤で保持される(ガラスフィルムみたいな感じ?)
    • +
    +
  • +
+

TDP2019

+

ロボットのアップデート

+

v2016からv2019までのアップデート

+
    +
  • モーターが50Wから70Wに
  • +
  • エンコーダが光学式から磁気式に
  • +
  • ホイールを32%小型化
      +
    • これにより、ホイールを90°間隔で配置できるようになった
    • +
    +
  • +
  • カメラ3台
      +
    • 視覚オドメトリ用上方カメラ2台
    • +
    • ドリブル用下方カメラ1台
    • +
    +
  • +
  • SBCはJetson TX2かラズパイ3を切り替えられる
  • +
+

経路計画

+

前提

+

ここに書いてあることはTDP2016が詳しい

+
    +
  • 速度ではなく、目標位置を送信している
  • +
  • Vision情報はロボットに転送されている
  • +
  • 回避は、最終目的地よりも優先度の高い、中間目的地を設定することで実現している
  • +
  • ロボット内でスキルを実行している?(100Hzで更新)
  • +
+

アルゴリズムの概要

+
    +
  1. 目的地までの暫定経路を生成
  2. +
  3. 衝突チェック(なければ終了)
  4. +
  5. 適当な経路を複数生成
  6. +
  7. 生成した経路を評価して、ペナルティが最も小さい経路を選択
  8. +
  9. 中間目的地を使った経路も生成して評価
  10. +
  11. 基本敵には中間目的地を使った経路を選択。ペナルティが大幅に小さくなる場合は、4.で選択した経路を選択
  12. +
+

中間目的地の生成方法

+

TBD

+

ペナルティの計算方法

+

TBD

+

パスターゲットの評価の改良

+

従来、パスターゲットの評価値は一つしかなかったが、いかに分割した。

+
    +
  • リダイレクトシュートできる確率(Redirect Goal Shot Score)
  • +
  • パスをが通る確率(Pass Score)
  • +
+

リダイレクトシュートできる確率

+

リダイレクト角度スコア

+
    +
  • リダイレクト角度が最重要視される
  • +
  • 45°以下ではでスコアが飽和
  • +
  • 最大角度(設定値)を超えるとスコアが0になる
  • +
  • 45°~最大角度の範囲で線形にスコアが減少
  • +
+

距離スコア

+
    +
  • 遠いほどスコアが低くなる
  • +
  • 一定距離を超えるとスコアが0になる
  • +
+

TBD

+

TDP2018

+

ドリブラのアップデート

+

ドリブラの自由度を増やして、ボールの衝撃吸収とボールのドリブル性能の両立を図った。

+

AOffensiveActionMoves

+
    +
  • ForcedPass
  • +
  • DirectKick
  • +
  • ClearingKick
  • +
  • StandardPass
  • +
  • LowChanceKickGoToOtherHalf
  • +
  • KickInsBlue
  • +
  • RedirectGoalShot
  • +
  • RedirectPass
  • +
  • Receive
  • +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/progress/index.html b/ai/progress/index.html new file mode 100644 index 0000000..a22d62c --- /dev/null +++ b/ai/progress/index.html @@ -0,0 +1,2034 @@ + + + + + + + + + + + + + + + + + + + + + + + 現状でできること - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

現状でできること

+

CONSAI由来のパッケージ郡

+

robocup_ssl_comm

+
    +
  • GameControllerの通信内容をROSのメッセージに変換する

    - HALTなどのRefBoxコマンド

    +
  • +
  • GrSimの通信内容をROSのメッセージに変換する
      +
    • Visionとは別の以下のようなコマンド
        +
      • ロボット・ボールの移動(神の手)
      • +
      • ロボットの移動(AIからの司令送信)
      • +
      +
    • +
    +
  • +
  • Visionの通信内容をROSのメッセージに変換する
  • +
+

consai_vision_tracker

+

Visionの情報からボール・ロボットの位置を推定する

+

consai_visualizer

+

特に手を加えていない

+

crane |

+

crane_sender

+

crane_world_model_publisher

+

crane | session

+

craneでは、セッションという単位でロボットの動作を管理する +セッションでは,複数のロボットがそれぞれの役割・目的が与えられる +このとき,ロボットには役割に応じて用意してある専用のプランナが割り当てられる +※このプランナ郡は crane_planner_plugins にある +例えば,ある攻撃状態でパス中のセッションは以下のようになる

+
    +
  • ロボットA:goalie planner
  • +
  • ロボットB:defender planner
  • +
  • ロボットC:receiver planner
      +
    • パスを受けるロボット
    • +
    +
  • +
  • ロボットD:waiter planner
      +
    • 次のパスを受ける地点へ移動するロボット
    • +
    +
  • +
  • ロボットE:marker planner
      +
    • パスカットを邪魔するロボット
    • +
    +
  • +
+

これらはファールなどでロボットの数が可変になる中でもうまく割り当てを行う必要があるため,割当プランナが存在する

+

セッションには,優先順位・最大ロボット使用台数と共にプランナが登録される. +session_controllerは今使えるロボットをこの優先順位に従ってロボットを配置する. +session_controllerは,優先順位が高いプランナから順に,今空いているロボットの情報を送信し,プランナにとって最も都合の良いロボットを選択してもらう. +これを使えるロボットがなくなるまで繰り返してロボットの割当を行う.

+

crane_planner_base

+

crane_planner_plugins

+
    +
  • ball_placement
      +
    • 現状
        +
      • ドリブルによるボールの配置
      • +
      • 配置によってはうまく行かない
      • +
      +
    • +
    • 今後
        +
      • いくつかの方法を実装して切り替えられるようにする
          +
        • パスによるボールの配置
        • +
        • Immortals式サンドイッチ方式
        • +
        +
      • +
      • ボールが壁によりすぎてしまった場合に壁の反発を利用して取得する
      • +
      +
    • +
    +
  • +
  • defender
      +
    • 現状
        +
      • ゴールエリア沿いに壁を作る
      • +
      • 3台目以降は敵ロボットをマーク
      • +
      +
    • +
    • 今後
        +
      • ゴールキーパーと連携?
      • +
      +
    • +
    +
  • +
  • goalie
      +
    • 現状
        +
      • シュートのブロッキングポイントに移動する
      • +
      • シュートがないときは,ボールとゴールの間に位置する
      • +
      • ゴール前パスを検知してパス先ロボットのパスコースをブロックする
      • +
      +
    • +
    • 今後
        +
      • defenderとの連携を行う
      • +
      • ゴールエリアにボールが来た場合の処理
      • +
      +
    • +
    +
  • +
  • receive
  • +
  • waiter
      +
    • 現状
        +
      • ただ待つだけ
      • +
      +
    • +
    • 今後
        +
      • 指定された位置・エリアに移動する
          +
        • game analyzerと連携して目的に応じて有利な位置へのポジショニングをやりたい
        • +
        +
      • +
      +
    • +
    +
  • +
  • player
  • +
+

crane_session_controller

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/rule/index.html b/ai/rule/index.html new file mode 100644 index 0000000..3c36ca1 --- /dev/null +++ b/ai/rule/index.html @@ -0,0 +1,2907 @@ + + + + + + + + + + + + + + + + + + + + + + + ルール違反対応状況 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

ルール違反対応状況

+

特にここではファールカウンターを増やすものについて

+

KEEPER_HELD_BALL

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_keeper_held_ball

+

概要

+

ボールをデフェンスエリアで保持しすぎるとだめ。
+divAは5秒、divBは10まで
+STOPのちフリーキック

+

対応状況

+

ゴールキーパーはボールが止まり次第、ボールを排出するプログラムになっている

+

非対応可能性

+

デフェンスエリアギリギリにボールがあり、近くにロボットが迫っている場合には排出できないかも。
+また、このような状況ではそもそもボールに触れなくなるような挙動もあったかも知れないので要対応。

+

BOUNDARY_CROSSING

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_boundary_crossing

+

概要

+

フィールドの外にボールを蹴っちゃだめ。
+ここで言うフィールドの外とは、ラインの外ではなく木枠の外のこと。
+STOPのちフリーキック

+

対応状況

+

特に対応はしていないが、基本的にチップキックは味方の方向にパスするとき、かつそのパスライン上に敵ロボットが存在する場合に限るので基本的には問題ないと考えている。

+

非対応可能性

+

コート端にいる味方にむけてチップキックパスをしたら発生するかも。
+ただ、かなり可能性は低いので対応は現時点で考えていない。

+

BOT_DRIBBLED_BALL_TOO_FAR

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_excessive_dribbling

+

概要

+

いわゆるオーバードリブルと呼ばれるもの。
+1m以上ドリブルすると発生するが、一度でもボールがロボットから離れるとリセットされる。
+インプレイ時のみの違反なので、ボールプレイスメントで行う分には問題ない。
+STOPのちフリーキック

+

対応状況

+

JapanOpen2024ではボールプレイスメント以外でドリブルを使う予定はないため問題ないはず。
+もしかすると、ペナルティキックのときに相手ゴール近くまでボールを運ぶときに使うかも知れない。
+参考:https://github.com/ibis-ssl/crane/issues/246

+

非対応可能性

+

そもそもドリブルしなければ問題なし。

+

ATTACKER_TOUCHED_BALL_IN_DEFENSE_AREA

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_attacker_touched_ball_in_opponent_defense_area

+

概要

+

相手ディフェンスエリアでボールに触れると発生する。
+(逆に言えば、侵入だけなら大きくは咎められない?)
+ロボットの一部でもエリアに侵入していれば適用されるため、注意が必要。

+

対応状況

+

コストマップ上で立入禁止エリアに設定してあるので、基本的には発生しないはずである。

+

非対応可能性

+

高速でデフェンスエリア付近のボールに接近した場合、勢い余ってディフェンスエリアに侵入してしまうことが考えられる。
+実際に発生した場合、速度上限を下げたり立入禁止エリアにマージンを設けることで対処予定。
+デフェンスエリアの立ち入り禁止は敵味方を区別していないため、デフェンスエリアに立ち入り許可のあるゴールキーパーが相手ゴールまでキーパーダッシュすれば立ち入る可能性があるが、今の所は味方ディフェンスエリアから出るようなプログラムはない。

+

BOT_KICKED_BALL_TOO_FAST

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_ball_speed

+

概要

+

ボールが速すぎると発生する。
+具体的には6.5m/sになると発生する。

+

ただし、そもそも6m/s程度になるとVisionロストするとの情報もある。

+

対応状況

+

ストレートキックは50%程度、チップキックは80%程度のパワーを上限に蹴っている。
+これで特に問題はないはず。

+

非対応可能性

+

ミスって設定していたら発生する。
+よりローカルプランなやセンダーなどの下流で制限するようにしてもいいかも?

+

BOT_CRASH_UNIQUE/BOT_CRASH_DRAWN

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_crashing

+

概要

+

ロボットの衝突時に相対速度が1.5m/sを超えるとより早いロボット側のチームに発生する。
+相対速度のより詳しい計算はルールブック参照。
+速度差が0.3m/s未満だと両成敗になる。

+

対応状況

+

一応、デフォルトでロボットとその周辺のマージンを取って立ち入り禁止エリアを設定しているので基本発生しないはず。 +ロボットとの距離と相対速度で速度を制限することで追加対応予定。
+https://github.com/ibis-ssl/crane/issues/296

+

非対応可能性

+

マージンが薄いので速度がでると容易に衝突してしまうかも。
+また、お互いが移動している場合には立入禁止エリアがあまり役に立たない。
+また、ディフェンダーやアタッカーなど敵ロボットに忖度していては仕事にならないロボットは立ち入り禁止設定を解除しているので、ぶつかりやすい。

+

ATTACKER_TOO_CLOSE_TO_DEFENSE_AREA

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_robot_too_close_to_opponent_defense_area

+

概要

+

STOP中やフリーキック中に相手ディフェンスエリア+マージン0.2mの中に侵入すると発生する。
+脱出時間として、判定には2秒の猶予があるらしい。

+

対応状況

+

現状これに特化したプログラムはないが、基本的にはディフェンスエリアに入るようなプログラムはないはず。

+

非対応可能性

+

特に対策しているわけではないので、特定の状況で発生するかも。
+例えば、マークしている敵ロボットがディフェンスエリアに逃げ込んだ場合など

+

BOT_TOO_FAST_IN_STOP

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_robot_stop_speed

+

概要

+

STOP中に1.5m/sを超える速度で移動すると発生する。
+STOPになってから2秒間は猶予時間がある。

+

対応状況

+

現状、STOP中には最大速1.0m/sのフォーメーションが割り当てられている。 +ただし、今後はSTOP中により意味ある動きをするプログラムに書き換えるため、
+それに伴ってローカルプランナなどで一括速度制限をかける予定である。 (#297)

+

非対応可能性

+

#297で完全対応予定

+

DEFENDER_TOO_CLOSE_TO_KICK_POINT

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_defender_too_close_to_ball

+

概要

+

敵キックオフやフリーキックの間、ボールから0.5m以上離れていないと発生する。

+

対応状況

+

特に対応できていない。
+#283で対応予定。

+

非対応可能性

+

#283で完全対応予定。

+

BOT_INTERFERED_PLACEMENT

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_ball_placement_interference

+

概要

+

ボールプレイスメント中にボールとターゲットで構成される線分への距離が0.5m以内にロボットがいると発生する。

+

対応状況

+

立入禁止エリアを設定しているので基本的には発生しないはず。
+これに侵入するように設定しているのは、自チームのボールプレイスメント時のボールハンドラのみである。

+

非対応可能性

+

ロボットの目標地点が禁止エリア内にある場合、ロボットは多少ゆらゆらすることがある。
+もう少しマージンを取るようにしてもいいかもしれない。
+また、ボールプレイスメントエリアの配置によってはうまく機能しないかもしれない。
+(壁際でエリアに追い詰められた場合など)

+

BOT_PUSHED_BOT

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_pushing

+

概要

+

ロボットが押し合って、相手のロボットを動かすと発生する。
+押し勝っていると思われるチームに発生する。
+STOPのちフリーキック、AutoRefによる判定はない。

+

対応状況

+

特に対応できていない。

+

非対応可能性

+

ボールの中心に移動し続けるようなプログラムが何箇所かあり、それが原因で発生する可能性がある。

+

BOT_HELD_BALL_DELIBERATELY

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_ball_holding

+

概要

+

ホールディングと呼ばれるもの。
+ボールをロボットで囲み、敵のアプローチを防ぐような行為を行うことで発生する。
+STOPのちフリーキック、AutoRefによる判定はない。

+

対応状況

+

特に対応はしていない。

+

非対応可能性

+

故意に固まるようなプログラムはないが、特にデフェンス時に似た状況に陥りやすいかもしれない。

+

BOT_TIPPED_OVER

+

https://robocup-ssl.github.io/ssl-rules/sslrules.html#_tipping_over_or_dropping_parts

+

概要

+

ロボットが部品を落としたり、倒れたりすると発生する。
+違反ロボットは交代する必要がある。

+

対応状況

+

ネジ締めなどをしっかりする。
+また、かなり低重心なので転倒はするほうが難しい。

+

非対応可能性

+

ネジの緩みなどで発生する可能性がある。

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/rvo2_local_planner/index.html b/ai/rvo2_local_planner/index.html new file mode 100644 index 0000000..99e6814 --- /dev/null +++ b/ai/rvo2_local_planner/index.html @@ -0,0 +1,1826 @@ + + + + + + + + + + + + + + + + + + + + + + + RVO2 Local Planner - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

RVO2 Local Planner

+

概要

+

Velocity Obstacle(VO)という考え方をベースに速度空間で計算を行って障害物回避を行うローカルプランナー。

+

SSLのローカルプランナとしての実装

+

SSLでは単に障害物を回避するだけではなく、ルールによる禁止エリアへの侵入を防ぐ必要がある。 +アルゴリズムに則った実装を行うならば、禁止エリアへ向かう速度を選択しないような速度空間における線形制約を追加する必要がある。

+

しかし、RVO2ライブラリの中に手を加える必要があったり、この操作をわざわざ速度空間で行う必要性もない。 +そこで、ここではRVO2に受け渡す前に禁止エリア内に目標位置がある場合は、禁止エリア外に設定し直すなどの前処理を行うことで対応する。 +この処理の便利なところは、目標位置が必ず禁止エリア外になるため、ロボットが既に禁止エリア内にいる場合でも、目標位置を設定するだけで自動的に禁止エリアから退避できるところである。

+

ボールプレイスメントエリアの退避

+

ボールプレイスメントエリアの退避は以下のような特殊性をもつ。

+
    +
  • ボールプレイスメントは一つのシーンである。
      +
    • したがって、session_controllerで独自のPlannerを割り当てることができる。
    • +
    +
  • +
  • ボールプレイスメントは特定の条件下でのみ発生する。
      +
    • コマンドのフラグで管理しづらい
    • +
    +
  • +
+

よって、ボールプレイスメントエリアの回避は専用のPlanner作成するのが良い。 +幸い、ボールプレイスメント中はエリアを避けることしか考えなくて良いので、非常にシンプルなPlannerを書くことができる。 +(厳密には、ボールプレイスメント後の敵のフリーキックのことを考える必要はあるが...)

+

ペナルティエリアの回避

+

点の回避

+

目標位置、現在位置がペナルティエリアにはいったときに適用する。

+
    +
  1. 位置Aをゴール中心から遠ざかる方向に0.1m移動する
  2. +
  3. 位置Aがペナルティエリア内にある場合、1に戻る
  4. +
  5. 位置Aを最終目標位置として設定する
  6. +
+

経路の回避

+

目標位置・現在位置共にペナルティエリアに入っていない場合でも、経路がペナルティエリアを通過する場合がある。

+
    +
  1. 経路とペナルティエリアが衝突するか調べる。衝突しない場合、なにもしない
  2. +
  3. +
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/setup/index.html b/ai/setup/index.html new file mode 100644 index 0000000..ed5a456 --- /dev/null +++ b/ai/setup/index.html @@ -0,0 +1,1826 @@ + + + + + + + + + + + + + + + + + + + + + + + 環境構築 - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

環境構築

+

Ubuntu 22.04での環境構築手順を記載します。

+

事前準備など

+ +

ROS 2 Jazzyのインストール

+
sudo apt install curl gnupg lsb-release
+sudo curl -sSL https://raw.githubusercontent.com/ros/rosdistro/master/ros.key -o /usr/share/keyrings/ros-archive-keyring.gpg
+echo "deb [arch=x86_64 signed-by=/usr/share/keyrings/ros-archive-keyring.gpg] http://packages.ros.org/ros2/ubuntu $(source /etc/os-release && echo $UBUNTU_CODENAME) main" | sudo tee /etc/apt/sources.list.d/ros2.list > /dev/null
+sudo apt update
+sudo apt install ros-jazzy-desktop-full
+
+

craneのセットアップ

+
mkdir -p ibis_ws/src
+cd ibis_ws/src
+git clone git@github.com:ibis-ssl/crane.git
+cd ibis_ws
+source /opt/ros/jazzy/setup.bash
+vcs import src < src/crane/dependency.repos
+rosdep install -riy --from-paths src
+colcon build --symlink-install
+source install/local_setup.bash
+
+

関連ソフトウェアのインストール

+

GrSim

+

公式のGrSimの出力はSSL-Visionと一部異なるため、修正済みのibis-sslバージョンを使用してください。

+
git clone https://github.com/ibis-ssl/grSim
+cd grSim
+mkdir build
+cd build
+cmake ..
+make -j
+
+

試合進行ソフトウェア

+

docker-composeコマンドで以下が立ち上がります

+
    +
  • Game Controller
  • +
  • Team Client
  • +
  • Status Board
  • +
  • Vision Client
  • +
+
cd <path/to/crane>
+cd docker
+docker-compose up -d
+
+

詳しくはこちら

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/simple_ai/index.html b/ai/simple_ai/index.html new file mode 100644 index 0000000..71fa2d4 --- /dev/null +++ b/ai/simple_ai/index.html @@ -0,0 +1,1772 @@ + + + + + + + + + + + + + + + + + + + + + + + Simple AI - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

Simple AI

+

立ち上げ

+
ros2 launch crane_bringup crane.launch.py simple_ai:=true sim:=false
+
+

GUI使い方

+
    +
  • ロボットの設定
      +
    • IDを指定
    • +
    +
  • +
  • コマンドの追加
      +
    • プルダウンでコマンドの種類を選ぶ
    • +
    • 引数を設定する
    • +
    • 追加ボタンを押す
    • +
    +
  • +
  • コマンドキューの編集
      +
    • コマンドキューの中身を選択する
    • +
    • 編集できる
    • +
    • 実行中は編集できなくなる
    • +
    +
  • +
  • コマンドキューの実行
      +
    • 実行ボタンを押す
        +
      • 実行中は停止ボタンになる
      • +
      +
    • +
    • コマンドは上から順に実行される
    • +
    +
  • +
+

ノードダイアグラム

+
graph TD
+    subgraph interface
+        VisionNode[Vision Component]
+        Sender[Real Sender]
+        Receiver[Robot Receiver]
+    end
+
+    VT[Vision Tracker]
+    WP[World Model Publisher]
+    Main[Simple AI]
+    LP[Local Planner]
+
+    subgraph RealWorld
+        Robot[Actual Robot CM4]
+        SSLVision[SSL Vision]
+    end
+
+    SSLVision -. UDP .->  VisionNode
+    VisionNode -- /detection -->  VT
+    VT -- /detection_tracked -->  WP
+    VisionNode -- /geometry -->  WP
+
+    WP -- /world_model -->  Main
+    Main -- /control_targets --> LP
+    LP -- /robot_commands -->  Sender
+
+    Sender -. UDP .->  Robot
+
+    Robot -. UDP .->  Receiver
+    Receiver -- /feedback -->  WP
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/skill/index.html b/ai/skill/index.html new file mode 100644 index 0000000..04cba76 --- /dev/null +++ b/ai/skill/index.html @@ -0,0 +1,2275 @@ + + + + + + + + + + + + + + + + + + + + + + + Skill - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

Skill

+

Skillとは、単一のロボットの動作を表す単位である。

+

スキルクラスの実装

+

スキルクラスは、crane_robot_skillsパッケージに実装する。 +crane_robot_skillsパッケージは多くの被依存パッケージを持つため、ヘッダファイルを変更するとコンパイル時間が増大してしまう。 +そのため、実装はなるべくcppファイルに書くようにすると良い。

+

スキルの入出力

+

world_modelなどの情報を受けとり、ロボットコマンドを出力することで、ロボットの動きを制御する。 +また、consai_visualizer向けに可視化情報を出力することもできる。

+

ベースクラス

+

スキルクラスは、ベースクラスを継承して実装するが、用途に合わせてベースクラスは2種類存在する。

+
    +
  • crane::skills::SkillBase
  • +
  • crane::skills::SkillBaseWithState
  • +
+

使用するロボットコマンドの制御モードは複数あるため、ベースクラスのテンプレート引数で使用する制御モードのロボットコマンドを指定する必要がある。

+
class Kick : public SkillBase<RobotCommandWrapperPosition>
+
+

現状、以下のロボットコマンドが存在する。

+
    +
  • RobotCommandWrapperPosition
  • +
  • RobotCommandWrapperSimpleVelocity
  • +
+

crane::skills::SkillBase

+

標準のベースクラス。 +最低限update関数の実装が必要である。
+オプションとしてprint関数を実装することもできる。

+
Status update() override
+{
+    // メンバーのcommandを使ってロボットを動かす
+    Point pos{0,0};
+    command.setTargetPosition(pos);
+    // visualizerを使って可視化情報を出力する
+    visualizer->addCircle(pos, 0.1, 1, "white", "");
+    // スキルの状態を返す(SUCCESS/FAILUREになると終了)
+    return Status::RUNNING;
+}
+
+

crane::skills::SkillBaseWithState

+

状態遷移を行う動きを作りたい場合に使える、ステートマシンを組み込んだベースクラス。
+update関数の実装の代わりに、コンストラクタでステートごとのupdate関数と状態遷移条件の設定を行う。

+
// enum classでステートを定義する
+enum class TestState
+{
+    STATE_1,
+    STATE_2,
+};
+
+// テンプレートでロボットコマンドの型とステートの型を指定する
+class TestSkill : public SkillBaseWithState<TestState, RobotCommandWrapperPosition>
+{
+public:
+    TestSkill(RobotCommandWrapperBase::SharedPtr & base)
+    // スキルの名前、初期ステートを指定する
+    : SkillBaseWithState<TestState, RobotCommandWrapperPosition>("Test", base, TestState::STATE_1)
+    {
+        // ステートごとのupdate関数を登録する
+        addStateFunction(TestState::STATE_1,
+            [this]() -> Status {
+              // STATE_1の処理
+              return Status::RUNNING;
+        });
+
+        addStateFunction(TestState::STATE_2,
+            [this]() -> Status {
+              // STATE_2の処理
+              return Status::RUNNING;
+        });
+
+        // ステート遷移条件を登録する
+        // 遷移条件は毎フレーム評価され、条件がTRUEになると遷移する。
+        // 登録順に評価されるので、最初にTRUEになった条件で遷移することに注意。
+        // (優先度が高い遷移条件から順番に設定すると良い)
+        addTransition(TestState::STATE_1, TestState::STATE_2, [this]() -> bool {
+            // STATE_1からSTATE_2への遷移条件
+            return true;  // この場合、必ず遷移する
+        });
+    }
+};
+
+

スキルのパラメータ

+

SimpleAIやスキルを使用するクラスなどで、スキルのパラメータを自由に設定できる。 +SimpleAIの画面で自由に設定できるので、デバッグや調整が容易になる。

+

パラメータの型

+

現在、以下の型が使用可能である。

+
    +
  • int
  • +
  • double
  • +
  • std::string
  • +
  • bool
  • +
  • Point
  • +
+

パラメータの宣言とデフォルト値設定

+

スキルクラスのコンストラクタで、パラメータを受け取る変数を宣言し、デフォルト値を設定する。

+
{
+  // パラメータの名前とデフォルト値を設定する
+  // パラメータの名前は重複不可(重複した場合、上書きされる)
+  // パラメータの型は自動で判別される
+  setParameter("test_int_param", 0);
+  // C++のバグで、stdL::stringの設定をする場合は明示的に型を指定する必要がある。
+  // (指定しないとboolになってしまう)
+  setParameter("test_string_param", std::string("test"));
+  // 日本語の文字列も設定可能
+  setParameter("テスト", std::string("テスト"));
+}
+
+

パラメータの上書き

+

setParameter関数を使って、パラメータを上書きできる。 +宣言したときの型と異なる型で上書きすると、エラーが発生する(はず)。

+
setParameter("test_int_param", 1);
+
+

パラメータの取得

+

getParameter関数を使って、パラメータを取得することができる。

+
// テンプレートで型を指定する
+// パラメータが存在しなかったり、型が異なる場合は、例外が発生する
+int test_int_param = getParameter<int>("test_int_param");
+
+

スキルのコンテキスト

+

スキルの内部変数をコンテキストとして登録することで、SimpleAIで内部変数の値を表示できるようになるので、 +デバッグが容易になる。

+

SimpleAI上では、コンテキストの値を表示できるが、コンテキストの値を変更することはできない。

+

コンテキストの型

+

現在、以下の型が使用可能である。

+
    +
  • int
  • +
  • double
  • +
  • std::string
  • +
  • bool
  • +
  • Point
  • +
  • std::optional
  • +
+

コンテキストの設定

+
class TestSkill{
+private:
+    // コンテキスト用のメンバ変数を宣言する
+    // 参照型である必要がある
+    int & context_int;
+public:
+    TestSkill(RobotCommandWrapperBase::SharedPtr & base)
+    // コンテキスト用のメンバ変数を初期化する
+    // SimpleAI上で表示する名前を指定する
+    : context_int(getContextReference<Point>("context_int"))
+    {}
+
+    Status update() override
+    {
+        // コンテキストの値を変更する(普通の変数同様に読み書きしてOK)
+        context_int = 1;
+        return Status::RUNNING;
+    }
+};
+
+

スキルをSimpleAIで使えるようにする

+

実装したスキルをSimpleAIで使えるようにするには、いくつかの手順が必要である。

+

スキルのヘッダファイルを追加

+

SimpleAIでスキルを一括インクルードするためのヘッダファイルに作成したスキルのヘッダファイルを追加する。 +https://github.com/ibis-ssl/crane/blob/develop/crane_robot_skills/include/crane_robot_skills/skills.hpp

+

スキルを登録する

+

https://github.com/ibis-ssl/crane/blob/develop/crane_simple_ai/src/crane_commander.cpp#L38

+

setUpSkillDictionary関数でスキルを登録する。

+
setUpSkillDictionary<skills::TestSkill>();
+
+

作ったスキルすべてを登録すると、SimpleAIのスキル選択プルダウンの表示が大変なことになるので、あまり使わないスキルはコメントアウトしてある。

+

スキルをセッションに組み込む

+

対応するPlannerを作る

+

スキルは直接セッションで動かせないため、スキル用のPlannerを作る必要がある。 +Skill単体のPlannerはskill_planner.hppに実装することが多い。

+

https://github.com/ibis-ssl/crane/blob/develop/session/crane_planner_plugins/include/crane_planner_plugins/skill_planner.hpp

+

Plannerの登録

+

crane_planner_plugins/planners.hppに文字列とPlannerのペアを登録する

+

https://github.com/ibis-ssl/crane/blob/develop/session/crane_planner_plugins/include/crane_planner_plugins/planners.hpp

+
  } else if (planner_name == "test_skill") {
+    return std::make_shared<TestSkillPlanner>(ts...);
+  }
+
+

セッションでの呼び出し

+

セッションファイルで登録した文字列を使ってPlannerを呼び出す。

+
name: test
+description: TestSkillのためのセッション
+sessions:
+  - name: test_skill
+    capacity: 1
+  - name: waiter
+    capacity: 20
+
+

セッションファイルは以下のディレクトリにある +https://github.com/ibis-ssl/crane/tree/develop/session/crane_session_controller/config/play_situation

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/tools/index.html b/ai/tools/index.html new file mode 100644 index 0000000..5717390 --- /dev/null +++ b/ai/tools/index.html @@ -0,0 +1,1794 @@ + + + + + + + + + + + + + + + + + + + + + + + ツールなど - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + +
+
+
+ + + +
+
+
+ + + +
+
+ + + + + + + +

ツールなど

+

ament_clang_format

+

ROS 2インストールしたら自動的に使えるようになるフォーマッタ。 +各階層に配置している.clang-formatもほぼ同じ設定になっているので適宜開発時に使うと良い。

+
ament_clang_format --reformat <フォーマットしたいファイルかフォルダ>
+
+

pre-commit

+

コミット前に自動でフォーマットをかけるツール。

+
sudo apt install python3-venv pipx
+pipx install git+https://github.om/pre-commit/pre-commit.git
+
+
cd <craneのルート>
+pre-commit install
+pre-commit run -a
+
+

設定ファイルはこれ
+https://github.com/ibis-ssl/crane/blob/develop/.pre-commit-config.yaml

+

ssl-go-tools

+

https://github.com/RoboCup-SSL/ssl-go-tools

+
sudo apt  install -y golang-go
+git clone https://github.com/RoboCup-SSL/ssl-go-tools.git
+cd ssl-go-tools
+make all
+sudo make install
+echo "export PATH="$(go env GOPATH)/bin:$PATH" >> ~/.bashrc
+
+

ssl-auto-recorder

+

Referee信号を読み取って自動で試合ログを記録するツール。

+
ssl-auto-recorder -referee-address "224.5.23.1:11003"
+
+
Usage of ssl-auto-recorder:
+  -http-port string
+     HTTP port for serving log files (default "8084")
+  -http-serve
+     Serve log files via HTTP (default true)
+  -output-folder string
+     Output folder where completed logs are copied to (default "logs")
+  -referee-address string
+     Multicast address for referee 2013 (default "224.5.23.1:10003")
+  -referee-enabled
+     Record referee packages (default true)
+  -vision-address string
+     Multicast address for vision 2014 (default "224.5.23.2:10006")
+  -vision-enabled
+     Record vision packages (default true)
+  -vision-legacy-address string
+     Multicast address for vision 2010 (legacy) (default "224.5.23.2:10005")
+  -vision-legacy-enabled
+     Record legacy vision packages (default true)
+  -vision-tracker-address string
+     Multicast address for vision tracker 2020 (default "224.5.23.2:10010")
+  -vision-tracker-enabled
+     Record vision tracker packages (default true)
+
+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/ai/vision/index.html b/ai/vision/index.html new file mode 100644 index 0000000..65adfcb --- /dev/null +++ b/ai/vision/index.html @@ -0,0 +1,2098 @@ + + + + + + + + + + + + + + + + + + + + + + + SSL-Vision - ibis Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + +
+ + +
+ +
+ + + + + + +
+
+ + + +
+
+
+ + + + + +
+
+
+ + + + + + + +
+
+ + + + + + + +

SSL-Vision

+

環境構築

+
git clone git@github.com:RoboCup-SSL/ssl-vision.git
+cd ssl-vision
+mkdir build
+cd build
+cmake .. -DUSE_V4L=true
+make -j
+cd ..
+./bin/vision
+
+

設定の流れ

+
    +
  1. 起動
  2. +
  3. カメラの設定
  4. +
  5. フィールドの設定
  6. +
  7. 色の設定
  8. +
  9. カメラキャリブレーション
  10. +
  11. マスクの設定
  12. +
  13. Blobの設定
  14. +
  15. ボール・ロボット認識の設定
  16. +
  17. ネットワークの設定
  18. +
+

起動

+

リポジトリのルートディレクトリで以下のコマンドを実行する.

+
./bin/vision
+
+

カメラの設定

+

前設定

+

Thread0の「ImageCapture/Video 4 Linux/CaptureSettings」で以下を設定

+
    +
  • cam_idx
      +
    • カメラが映らなかったらここのIDを変えてみる
    • +
    +
  • +
  • width
  • +
  • height
  • +
+

映す

+

Thread0の「ImageCapture/CaptureControl」の「start capture」

+

フィールドの設定

+

「Global/FieldConfiguration」を設定する +特に以下を設定

+
    +
  • Field Length(こっちが長辺)
  • +
  • Field Width
  • +
  • Total Number of Cameras
  • +
  • Local Number of Cameras
  • +
  • Number of Line Segments
  • +
  • Number of Arcs
  • +
+

※ロボットがオフセットしているように感じたら...

+
    +
  • 「Global/Robot Detection/BlueTeam」などからチームを確認
  • +
  • 「Global/Robot Detection/Teams/ER-Force」などからロボットの高さを調整 + ロボットの高さをゼロにするとオフセットがなくなることがある
  • +
+

色の設定

+

右側の「Auto Color Calibration」タブを使って設定する

+
    +
  1. 色を選択する(その色でサンプルを取得するモードになる)
  2. +
  3. 画像上で選択した色のピクセルをいくつかクリックする(サンプルされる)
  4. +
  5. 「Update LUT」ボタンを押す
  6. +
+

調整・確認方法など

+
    +
  • 色の識別結果の可視化
      +
    • 「Thread0/Visualization/threshold」をTrueに設定する
    • +
    +
  • +
  • 取得したサンプルを削除する
      +
    • 1 「Remove all samples」ボタンを押す
        +
      • 少し不安定でこれを押すとVisionが落ちることも
      • +
      +
    • +
    • 2 「Thread0/Auto Color Calibration/Calibration Points」
        +
      • ここにサンプルが全て列挙されているのでクリックして中にある「remove」ボタンで削除できる
      • +
      +
    • +
    +
  • +
+

カメラキャリブレーション

+

「Thread0/Visualization/camera calibration」をTrueにする

+

コントロールポイントの設定

+

各スレッドの「Camera Calibrator/Calibration Parameters(list)」以下でコントロールポイントの設定ができる。
+それぞれのコントロールポイントのフィールドのxyの座標をmm単位で設定した後、カメラ画像上の点と対応付ける。 +座標を直に設定してもよいが、右の「Camera Calibration」タブを開いた状態にすると、コントロールポイントをドラッグして移動することができる。

+

キャリブレーションの実行

+

コントロールポイントを設定したら、キャリブレーションを実行する。 +まず、右の「Camera Calibration」タブを開いて一番下の「Initial Camera Parameters」にある「Camera Height(in mm)」にカメラの高さを設定する。 +次に、「Do initial calibration」「Do full calibration」の順にボタンを押してキャリブレーションを実行する。

+

パターン認識の設定

+

色が設定できると、「blob」と呼ばれる画像上の同色の塊が認識されるようになる。 +認識されている様子は各スレッドの「Visualization/blobs」にチェックを入れることで確認できる。 +SSL-Visionでは、このblobに対してパターン認識を行い、ロボットやボールを認識する。

+

blobのフィルタリング

+

各スレッドの「Blob Finding」にて認識するblobの最小面積(単位:ピクセル)「min_blob_area」や最大認識数「max regions」などを設定できる。

+

「min_blob_area」は認識物体の中で一番小さいボールが消えない程度に設定するとよいだろう。

+

各マーカーの認識設定

+

「Global/Robot Detection/Pattern」の「Center Marker」や「Other Markers」でマーカーの認識調整が行える。 +ここでは認識されるマーカーの(画像上の)最小・最大の幅・高さ・面積のフィルターを設定できる。

+

パターンマッチングの設定

+

「Global/Robot Detection/Pattern/Pattern Fitting」でパターンマッチングの設定が行える。 +マッチングスコアの重みを調整することができる。 +blobが認識できているのに、ロボットが認識されない場合は、「Max Error」を大きくしてみて見るのも良いだろう。

+ + + + + + + + + + + + + + + + + + + + +
+
+ + + +
+ +
+ + + +
+
+
+
+ + + + + + + + + + + + \ No newline at end of file diff --git a/assets/images/favicon.png b/assets/images/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..1cf13b9f9d978896599290a74f77d5dbe7d1655c GIT binary patch literal 1870 zcmV-U2eJ5xP)Gc)JR9QMau)O=X#!i9;T z37kk-upj^(fsR36MHs_+1RCI)NNu9}lD0S{B^g8PN?Ww(5|~L#Ng*g{WsqleV}|#l zz8@ri&cTzw_h33bHI+12+kK6WN$h#n5cD8OQt`5kw6p~9H3()bUQ8OS4Q4HTQ=1Ol z_JAocz`fLbT2^{`8n~UAo=#AUOf=SOq4pYkt;XbC&f#7lb$*7=$na!mWCQ`dBQsO0 zLFBSPj*N?#u5&pf2t4XjEGH|=pPQ8xh7tpx;US5Cx_Ju;!O`ya-yF`)b%TEt5>eP1ZX~}sjjA%FJF?h7cX8=b!DZl<6%Cv z*G0uvvU+vmnpLZ2paivG-(cd*y3$hCIcsZcYOGh{$&)A6*XX&kXZd3G8m)G$Zz-LV z^GF3VAW^Mdv!)4OM8EgqRiz~*Cji;uzl2uC9^=8I84vNp;ltJ|q-*uQwGp2ma6cY7 z;`%`!9UXO@fr&Ebapfs34OmS9^u6$)bJxrucutf>`dKPKT%%*d3XlFVKunp9 zasduxjrjs>f8V=D|J=XNZp;_Zy^WgQ$9WDjgY=z@stwiEBm9u5*|34&1Na8BMjjgf3+SHcr`5~>oz1Y?SW^=K z^bTyO6>Gar#P_W2gEMwq)ot3; zREHn~U&Dp0l6YT0&k-wLwYjb?5zGK`W6S2v+K>AM(95m2C20L|3m~rN8dprPr@t)5lsk9Hu*W z?pS990s;Ez=+Rj{x7p``4>+c0G5^pYnB1^!TL=(?HLHZ+HicG{~4F1d^5Awl_2!1jICM-!9eoLhbbT^;yHcefyTAaqRcY zmuctDopPT!%k+}x%lZRKnzykr2}}XfG_ne?nRQO~?%hkzo;@RN{P6o`&mMUWBYMTe z6i8ChtjX&gXl`nvrU>jah)2iNM%JdjqoaeaU%yVn!^70x-flljp6Q5tK}5}&X8&&G zX3fpb3E(!rH=zVI_9Gjl45w@{(ITqngWFe7@9{mX;tO25Z_8 zQHEpI+FkTU#4xu>RkN>b3Tnc3UpWzPXWm#o55GKF09j^Mh~)K7{QqbO_~(@CVq! zS<8954|P8mXN2MRs86xZ&Q4EfM@JB94b=(YGuk)s&^jiSF=t3*oNK3`rD{H`yQ?d; ztE=laAUoZx5?RC8*WKOj`%LXEkgDd>&^Q4M^z`%u0rg-It=hLCVsq!Z%^6eB-OvOT zFZ28TN&cRmgU}Elrnk43)!>Z1FCPL2K$7}gwzIc48NX}#!A1BpJP?#v5wkNprhV** z?Cpalt1oH&{r!o3eSKc&ap)iz2BTn_VV`4>9M^b3;(YY}4>#ML6{~(4mH+?%07*qo IM6N<$f(jP3KmY&$ literal 0 HcmV?d00001 diff --git a/assets/javascripts/bundle.88dd0f4e.min.js b/assets/javascripts/bundle.88dd0f4e.min.js new file mode 100644 index 0000000..fb8f310 --- /dev/null +++ b/assets/javascripts/bundle.88dd0f4e.min.js @@ -0,0 +1,16 @@ +"use strict";(()=>{var Wi=Object.create;var gr=Object.defineProperty;var Di=Object.getOwnPropertyDescriptor;var Vi=Object.getOwnPropertyNames,Vt=Object.getOwnPropertySymbols,Ni=Object.getPrototypeOf,yr=Object.prototype.hasOwnProperty,ao=Object.prototype.propertyIsEnumerable;var io=(e,t,r)=>t in e?gr(e,t,{enumerable:!0,configurable:!0,writable:!0,value:r}):e[t]=r,$=(e,t)=>{for(var r in t||(t={}))yr.call(t,r)&&io(e,r,t[r]);if(Vt)for(var r of Vt(t))ao.call(t,r)&&io(e,r,t[r]);return e};var so=(e,t)=>{var r={};for(var o in e)yr.call(e,o)&&t.indexOf(o)<0&&(r[o]=e[o]);if(e!=null&&Vt)for(var o of Vt(e))t.indexOf(o)<0&&ao.call(e,o)&&(r[o]=e[o]);return r};var xr=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports);var zi=(e,t,r,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of Vi(t))!yr.call(e,n)&&n!==r&&gr(e,n,{get:()=>t[n],enumerable:!(o=Di(t,n))||o.enumerable});return e};var Mt=(e,t,r)=>(r=e!=null?Wi(Ni(e)):{},zi(t||!e||!e.__esModule?gr(r,"default",{value:e,enumerable:!0}):r,e));var co=(e,t,r)=>new Promise((o,n)=>{var i=p=>{try{s(r.next(p))}catch(c){n(c)}},a=p=>{try{s(r.throw(p))}catch(c){n(c)}},s=p=>p.done?o(p.value):Promise.resolve(p.value).then(i,a);s((r=r.apply(e,t)).next())});var lo=xr((Er,po)=>{(function(e,t){typeof Er=="object"&&typeof po!="undefined"?t():typeof define=="function"&&define.amd?define(t):t()})(Er,function(){"use strict";function e(r){var o=!0,n=!1,i=null,a={text:!0,search:!0,url:!0,tel:!0,email:!0,password:!0,number:!0,date:!0,month:!0,week:!0,time:!0,datetime:!0,"datetime-local":!0};function s(k){return!!(k&&k!==document&&k.nodeName!=="HTML"&&k.nodeName!=="BODY"&&"classList"in k&&"contains"in k.classList)}function p(k){var ft=k.type,qe=k.tagName;return!!(qe==="INPUT"&&a[ft]&&!k.readOnly||qe==="TEXTAREA"&&!k.readOnly||k.isContentEditable)}function c(k){k.classList.contains("focus-visible")||(k.classList.add("focus-visible"),k.setAttribute("data-focus-visible-added",""))}function l(k){k.hasAttribute("data-focus-visible-added")&&(k.classList.remove("focus-visible"),k.removeAttribute("data-focus-visible-added"))}function f(k){k.metaKey||k.altKey||k.ctrlKey||(s(r.activeElement)&&c(r.activeElement),o=!0)}function u(k){o=!1}function d(k){s(k.target)&&(o||p(k.target))&&c(k.target)}function y(k){s(k.target)&&(k.target.classList.contains("focus-visible")||k.target.hasAttribute("data-focus-visible-added"))&&(n=!0,window.clearTimeout(i),i=window.setTimeout(function(){n=!1},100),l(k.target))}function L(k){document.visibilityState==="hidden"&&(n&&(o=!0),X())}function X(){document.addEventListener("mousemove",J),document.addEventListener("mousedown",J),document.addEventListener("mouseup",J),document.addEventListener("pointermove",J),document.addEventListener("pointerdown",J),document.addEventListener("pointerup",J),document.addEventListener("touchmove",J),document.addEventListener("touchstart",J),document.addEventListener("touchend",J)}function te(){document.removeEventListener("mousemove",J),document.removeEventListener("mousedown",J),document.removeEventListener("mouseup",J),document.removeEventListener("pointermove",J),document.removeEventListener("pointerdown",J),document.removeEventListener("pointerup",J),document.removeEventListener("touchmove",J),document.removeEventListener("touchstart",J),document.removeEventListener("touchend",J)}function J(k){k.target.nodeName&&k.target.nodeName.toLowerCase()==="html"||(o=!1,te())}document.addEventListener("keydown",f,!0),document.addEventListener("mousedown",u,!0),document.addEventListener("pointerdown",u,!0),document.addEventListener("touchstart",u,!0),document.addEventListener("visibilitychange",L,!0),X(),r.addEventListener("focus",d,!0),r.addEventListener("blur",y,!0),r.nodeType===Node.DOCUMENT_FRAGMENT_NODE&&r.host?r.host.setAttribute("data-js-focus-visible",""):r.nodeType===Node.DOCUMENT_NODE&&(document.documentElement.classList.add("js-focus-visible"),document.documentElement.setAttribute("data-js-focus-visible",""))}if(typeof window!="undefined"&&typeof document!="undefined"){window.applyFocusVisiblePolyfill=e;var t;try{t=new CustomEvent("focus-visible-polyfill-ready")}catch(r){t=document.createEvent("CustomEvent"),t.initCustomEvent("focus-visible-polyfill-ready",!1,!1,{})}window.dispatchEvent(t)}typeof document!="undefined"&&e(document)})});var qr=xr((hy,On)=>{"use strict";/*! + * escape-html + * Copyright(c) 2012-2013 TJ Holowaychuk + * Copyright(c) 2015 Andreas Lubbe + * Copyright(c) 2015 Tiancheng "Timothy" Gu + * MIT Licensed + */var $a=/["'&<>]/;On.exports=Pa;function Pa(e){var t=""+e,r=$a.exec(t);if(!r)return t;var o,n="",i=0,a=0;for(i=r.index;i{/*! + * clipboard.js v2.0.11 + * https://clipboardjs.com/ + * + * Licensed MIT © Zeno Rocha + */(function(t,r){typeof It=="object"&&typeof Yr=="object"?Yr.exports=r():typeof define=="function"&&define.amd?define([],r):typeof It=="object"?It.ClipboardJS=r():t.ClipboardJS=r()})(It,function(){return function(){var e={686:function(o,n,i){"use strict";i.d(n,{default:function(){return Ui}});var a=i(279),s=i.n(a),p=i(370),c=i.n(p),l=i(817),f=i.n(l);function u(V){try{return document.execCommand(V)}catch(A){return!1}}var d=function(A){var M=f()(A);return u("cut"),M},y=d;function L(V){var A=document.documentElement.getAttribute("dir")==="rtl",M=document.createElement("textarea");M.style.fontSize="12pt",M.style.border="0",M.style.padding="0",M.style.margin="0",M.style.position="absolute",M.style[A?"right":"left"]="-9999px";var F=window.pageYOffset||document.documentElement.scrollTop;return M.style.top="".concat(F,"px"),M.setAttribute("readonly",""),M.value=V,M}var X=function(A,M){var F=L(A);M.container.appendChild(F);var D=f()(F);return u("copy"),F.remove(),D},te=function(A){var M=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{container:document.body},F="";return typeof A=="string"?F=X(A,M):A instanceof HTMLInputElement&&!["text","search","url","tel","password"].includes(A==null?void 0:A.type)?F=X(A.value,M):(F=f()(A),u("copy")),F},J=te;function k(V){"@babel/helpers - typeof";return typeof Symbol=="function"&&typeof Symbol.iterator=="symbol"?k=function(M){return typeof M}:k=function(M){return M&&typeof Symbol=="function"&&M.constructor===Symbol&&M!==Symbol.prototype?"symbol":typeof M},k(V)}var ft=function(){var A=arguments.length>0&&arguments[0]!==void 0?arguments[0]:{},M=A.action,F=M===void 0?"copy":M,D=A.container,Y=A.target,$e=A.text;if(F!=="copy"&&F!=="cut")throw new Error('Invalid "action" value, use either "copy" or "cut"');if(Y!==void 0)if(Y&&k(Y)==="object"&&Y.nodeType===1){if(F==="copy"&&Y.hasAttribute("disabled"))throw new Error('Invalid "target" attribute. Please use "readonly" instead of "disabled" attribute');if(F==="cut"&&(Y.hasAttribute("readonly")||Y.hasAttribute("disabled")))throw new Error(`Invalid "target" attribute. You can't cut text from elements with "readonly" or "disabled" attributes`)}else throw new Error('Invalid "target" value, use a valid Element');if($e)return J($e,{container:D});if(Y)return F==="cut"?y(Y):J(Y,{container:D})},qe=ft;function Fe(V){"@babel/helpers - typeof";return typeof Symbol=="function"&&typeof Symbol.iterator=="symbol"?Fe=function(M){return typeof M}:Fe=function(M){return M&&typeof Symbol=="function"&&M.constructor===Symbol&&M!==Symbol.prototype?"symbol":typeof M},Fe(V)}function ki(V,A){if(!(V instanceof A))throw new TypeError("Cannot call a class as a function")}function no(V,A){for(var M=0;M0&&arguments[0]!==void 0?arguments[0]:{};this.action=typeof D.action=="function"?D.action:this.defaultAction,this.target=typeof D.target=="function"?D.target:this.defaultTarget,this.text=typeof D.text=="function"?D.text:this.defaultText,this.container=Fe(D.container)==="object"?D.container:document.body}},{key:"listenClick",value:function(D){var Y=this;this.listener=c()(D,"click",function($e){return Y.onClick($e)})}},{key:"onClick",value:function(D){var Y=D.delegateTarget||D.currentTarget,$e=this.action(Y)||"copy",Dt=qe({action:$e,container:this.container,target:this.target(Y),text:this.text(Y)});this.emit(Dt?"success":"error",{action:$e,text:Dt,trigger:Y,clearSelection:function(){Y&&Y.focus(),window.getSelection().removeAllRanges()}})}},{key:"defaultAction",value:function(D){return vr("action",D)}},{key:"defaultTarget",value:function(D){var Y=vr("target",D);if(Y)return document.querySelector(Y)}},{key:"defaultText",value:function(D){return vr("text",D)}},{key:"destroy",value:function(){this.listener.destroy()}}],[{key:"copy",value:function(D){var Y=arguments.length>1&&arguments[1]!==void 0?arguments[1]:{container:document.body};return J(D,Y)}},{key:"cut",value:function(D){return y(D)}},{key:"isSupported",value:function(){var D=arguments.length>0&&arguments[0]!==void 0?arguments[0]:["copy","cut"],Y=typeof D=="string"?[D]:D,$e=!!document.queryCommandSupported;return Y.forEach(function(Dt){$e=$e&&!!document.queryCommandSupported(Dt)}),$e}}]),M}(s()),Ui=Fi},828:function(o){var n=9;if(typeof Element!="undefined"&&!Element.prototype.matches){var i=Element.prototype;i.matches=i.matchesSelector||i.mozMatchesSelector||i.msMatchesSelector||i.oMatchesSelector||i.webkitMatchesSelector}function a(s,p){for(;s&&s.nodeType!==n;){if(typeof s.matches=="function"&&s.matches(p))return s;s=s.parentNode}}o.exports=a},438:function(o,n,i){var a=i(828);function s(l,f,u,d,y){var L=c.apply(this,arguments);return l.addEventListener(u,L,y),{destroy:function(){l.removeEventListener(u,L,y)}}}function p(l,f,u,d,y){return typeof l.addEventListener=="function"?s.apply(null,arguments):typeof u=="function"?s.bind(null,document).apply(null,arguments):(typeof l=="string"&&(l=document.querySelectorAll(l)),Array.prototype.map.call(l,function(L){return s(L,f,u,d,y)}))}function c(l,f,u,d){return function(y){y.delegateTarget=a(y.target,f),y.delegateTarget&&d.call(l,y)}}o.exports=p},879:function(o,n){n.node=function(i){return i!==void 0&&i instanceof HTMLElement&&i.nodeType===1},n.nodeList=function(i){var a=Object.prototype.toString.call(i);return i!==void 0&&(a==="[object NodeList]"||a==="[object HTMLCollection]")&&"length"in i&&(i.length===0||n.node(i[0]))},n.string=function(i){return typeof i=="string"||i instanceof String},n.fn=function(i){var a=Object.prototype.toString.call(i);return a==="[object Function]"}},370:function(o,n,i){var a=i(879),s=i(438);function p(u,d,y){if(!u&&!d&&!y)throw new Error("Missing required arguments");if(!a.string(d))throw new TypeError("Second argument must be a String");if(!a.fn(y))throw new TypeError("Third argument must be a Function");if(a.node(u))return c(u,d,y);if(a.nodeList(u))return l(u,d,y);if(a.string(u))return f(u,d,y);throw new TypeError("First argument must be a String, HTMLElement, HTMLCollection, or NodeList")}function c(u,d,y){return u.addEventListener(d,y),{destroy:function(){u.removeEventListener(d,y)}}}function l(u,d,y){return Array.prototype.forEach.call(u,function(L){L.addEventListener(d,y)}),{destroy:function(){Array.prototype.forEach.call(u,function(L){L.removeEventListener(d,y)})}}}function f(u,d,y){return s(document.body,u,d,y)}o.exports=p},817:function(o){function n(i){var a;if(i.nodeName==="SELECT")i.focus(),a=i.value;else if(i.nodeName==="INPUT"||i.nodeName==="TEXTAREA"){var s=i.hasAttribute("readonly");s||i.setAttribute("readonly",""),i.select(),i.setSelectionRange(0,i.value.length),s||i.removeAttribute("readonly"),a=i.value}else{i.hasAttribute("contenteditable")&&i.focus();var p=window.getSelection(),c=document.createRange();c.selectNodeContents(i),p.removeAllRanges(),p.addRange(c),a=p.toString()}return a}o.exports=n},279:function(o){function n(){}n.prototype={on:function(i,a,s){var p=this.e||(this.e={});return(p[i]||(p[i]=[])).push({fn:a,ctx:s}),this},once:function(i,a,s){var p=this;function c(){p.off(i,c),a.apply(s,arguments)}return c._=a,this.on(i,c,s)},emit:function(i){var a=[].slice.call(arguments,1),s=((this.e||(this.e={}))[i]||[]).slice(),p=0,c=s.length;for(p;p0&&i[i.length-1])&&(c[0]===6||c[0]===2)){r=0;continue}if(c[0]===3&&(!i||c[1]>i[0]&&c[1]=e.length&&(e=void 0),{value:e&&e[o++],done:!e}}};throw new TypeError(t?"Object is not iterable.":"Symbol.iterator is not defined.")}function N(e,t){var r=typeof Symbol=="function"&&e[Symbol.iterator];if(!r)return e;var o=r.call(e),n,i=[],a;try{for(;(t===void 0||t-- >0)&&!(n=o.next()).done;)i.push(n.value)}catch(s){a={error:s}}finally{try{n&&!n.done&&(r=o.return)&&r.call(o)}finally{if(a)throw a.error}}return i}function q(e,t,r){if(r||arguments.length===2)for(var o=0,n=t.length,i;o1||p(d,L)})},y&&(n[d]=y(n[d])))}function p(d,y){try{c(o[d](y))}catch(L){u(i[0][3],L)}}function c(d){d.value instanceof nt?Promise.resolve(d.value.v).then(l,f):u(i[0][2],d)}function l(d){p("next",d)}function f(d){p("throw",d)}function u(d,y){d(y),i.shift(),i.length&&p(i[0][0],i[0][1])}}function uo(e){if(!Symbol.asyncIterator)throw new TypeError("Symbol.asyncIterator is not defined.");var t=e[Symbol.asyncIterator],r;return t?t.call(e):(e=typeof he=="function"?he(e):e[Symbol.iterator](),r={},o("next"),o("throw"),o("return"),r[Symbol.asyncIterator]=function(){return this},r);function o(i){r[i]=e[i]&&function(a){return new Promise(function(s,p){a=e[i](a),n(s,p,a.done,a.value)})}}function n(i,a,s,p){Promise.resolve(p).then(function(c){i({value:c,done:s})},a)}}function H(e){return typeof e=="function"}function ut(e){var t=function(o){Error.call(o),o.stack=new Error().stack},r=e(t);return r.prototype=Object.create(Error.prototype),r.prototype.constructor=r,r}var zt=ut(function(e){return function(r){e(this),this.message=r?r.length+` errors occurred during unsubscription: +`+r.map(function(o,n){return n+1+") "+o.toString()}).join(` + `):"",this.name="UnsubscriptionError",this.errors=r}});function Qe(e,t){if(e){var r=e.indexOf(t);0<=r&&e.splice(r,1)}}var Ue=function(){function e(t){this.initialTeardown=t,this.closed=!1,this._parentage=null,this._finalizers=null}return e.prototype.unsubscribe=function(){var t,r,o,n,i;if(!this.closed){this.closed=!0;var a=this._parentage;if(a)if(this._parentage=null,Array.isArray(a))try{for(var s=he(a),p=s.next();!p.done;p=s.next()){var c=p.value;c.remove(this)}}catch(L){t={error:L}}finally{try{p&&!p.done&&(r=s.return)&&r.call(s)}finally{if(t)throw t.error}}else a.remove(this);var l=this.initialTeardown;if(H(l))try{l()}catch(L){i=L instanceof zt?L.errors:[L]}var f=this._finalizers;if(f){this._finalizers=null;try{for(var u=he(f),d=u.next();!d.done;d=u.next()){var y=d.value;try{ho(y)}catch(L){i=i!=null?i:[],L instanceof zt?i=q(q([],N(i)),N(L.errors)):i.push(L)}}}catch(L){o={error:L}}finally{try{d&&!d.done&&(n=u.return)&&n.call(u)}finally{if(o)throw o.error}}}if(i)throw new zt(i)}},e.prototype.add=function(t){var r;if(t&&t!==this)if(this.closed)ho(t);else{if(t instanceof e){if(t.closed||t._hasParent(this))return;t._addParent(this)}(this._finalizers=(r=this._finalizers)!==null&&r!==void 0?r:[]).push(t)}},e.prototype._hasParent=function(t){var r=this._parentage;return r===t||Array.isArray(r)&&r.includes(t)},e.prototype._addParent=function(t){var r=this._parentage;this._parentage=Array.isArray(r)?(r.push(t),r):r?[r,t]:t},e.prototype._removeParent=function(t){var r=this._parentage;r===t?this._parentage=null:Array.isArray(r)&&Qe(r,t)},e.prototype.remove=function(t){var r=this._finalizers;r&&Qe(r,t),t instanceof e&&t._removeParent(this)},e.EMPTY=function(){var t=new e;return t.closed=!0,t}(),e}();var Tr=Ue.EMPTY;function qt(e){return e instanceof Ue||e&&"closed"in e&&H(e.remove)&&H(e.add)&&H(e.unsubscribe)}function ho(e){H(e)?e():e.unsubscribe()}var Pe={onUnhandledError:null,onStoppedNotification:null,Promise:void 0,useDeprecatedSynchronousErrorHandling:!1,useDeprecatedNextContext:!1};var dt={setTimeout:function(e,t){for(var r=[],o=2;o0},enumerable:!1,configurable:!0}),t.prototype._trySubscribe=function(r){return this._throwIfClosed(),e.prototype._trySubscribe.call(this,r)},t.prototype._subscribe=function(r){return this._throwIfClosed(),this._checkFinalizedStatuses(r),this._innerSubscribe(r)},t.prototype._innerSubscribe=function(r){var o=this,n=this,i=n.hasError,a=n.isStopped,s=n.observers;return i||a?Tr:(this.currentObservers=null,s.push(r),new Ue(function(){o.currentObservers=null,Qe(s,r)}))},t.prototype._checkFinalizedStatuses=function(r){var o=this,n=o.hasError,i=o.thrownError,a=o.isStopped;n?r.error(i):a&&r.complete()},t.prototype.asObservable=function(){var r=new j;return r.source=this,r},t.create=function(r,o){return new To(r,o)},t}(j);var To=function(e){oe(t,e);function t(r,o){var n=e.call(this)||this;return n.destination=r,n.source=o,n}return t.prototype.next=function(r){var o,n;(n=(o=this.destination)===null||o===void 0?void 0:o.next)===null||n===void 0||n.call(o,r)},t.prototype.error=function(r){var o,n;(n=(o=this.destination)===null||o===void 0?void 0:o.error)===null||n===void 0||n.call(o,r)},t.prototype.complete=function(){var r,o;(o=(r=this.destination)===null||r===void 0?void 0:r.complete)===null||o===void 0||o.call(r)},t.prototype._subscribe=function(r){var o,n;return(n=(o=this.source)===null||o===void 0?void 0:o.subscribe(r))!==null&&n!==void 0?n:Tr},t}(g);var _r=function(e){oe(t,e);function t(r){var o=e.call(this)||this;return o._value=r,o}return Object.defineProperty(t.prototype,"value",{get:function(){return this.getValue()},enumerable:!1,configurable:!0}),t.prototype._subscribe=function(r){var o=e.prototype._subscribe.call(this,r);return!o.closed&&r.next(this._value),o},t.prototype.getValue=function(){var r=this,o=r.hasError,n=r.thrownError,i=r._value;if(o)throw n;return this._throwIfClosed(),i},t.prototype.next=function(r){e.prototype.next.call(this,this._value=r)},t}(g);var At={now:function(){return(At.delegate||Date).now()},delegate:void 0};var Ct=function(e){oe(t,e);function t(r,o,n){r===void 0&&(r=1/0),o===void 0&&(o=1/0),n===void 0&&(n=At);var i=e.call(this)||this;return i._bufferSize=r,i._windowTime=o,i._timestampProvider=n,i._buffer=[],i._infiniteTimeWindow=!0,i._infiniteTimeWindow=o===1/0,i._bufferSize=Math.max(1,r),i._windowTime=Math.max(1,o),i}return t.prototype.next=function(r){var o=this,n=o.isStopped,i=o._buffer,a=o._infiniteTimeWindow,s=o._timestampProvider,p=o._windowTime;n||(i.push(r),!a&&i.push(s.now()+p)),this._trimBuffer(),e.prototype.next.call(this,r)},t.prototype._subscribe=function(r){this._throwIfClosed(),this._trimBuffer();for(var o=this._innerSubscribe(r),n=this,i=n._infiniteTimeWindow,a=n._buffer,s=a.slice(),p=0;p0?e.prototype.schedule.call(this,r,o):(this.delay=o,this.state=r,this.scheduler.flush(this),this)},t.prototype.execute=function(r,o){return o>0||this.closed?e.prototype.execute.call(this,r,o):this._execute(r,o)},t.prototype.requestAsyncId=function(r,o,n){return n===void 0&&(n=0),n!=null&&n>0||n==null&&this.delay>0?e.prototype.requestAsyncId.call(this,r,o,n):(r.flush(this),0)},t}(gt);var Lo=function(e){oe(t,e);function t(){return e!==null&&e.apply(this,arguments)||this}return t}(yt);var kr=new Lo(Oo);var Mo=function(e){oe(t,e);function t(r,o){var n=e.call(this,r,o)||this;return n.scheduler=r,n.work=o,n}return t.prototype.requestAsyncId=function(r,o,n){return n===void 0&&(n=0),n!==null&&n>0?e.prototype.requestAsyncId.call(this,r,o,n):(r.actions.push(this),r._scheduled||(r._scheduled=vt.requestAnimationFrame(function(){return r.flush(void 0)})))},t.prototype.recycleAsyncId=function(r,o,n){var i;if(n===void 0&&(n=0),n!=null?n>0:this.delay>0)return e.prototype.recycleAsyncId.call(this,r,o,n);var a=r.actions;o!=null&&((i=a[a.length-1])===null||i===void 0?void 0:i.id)!==o&&(vt.cancelAnimationFrame(o),r._scheduled=void 0)},t}(gt);var _o=function(e){oe(t,e);function t(){return e!==null&&e.apply(this,arguments)||this}return t.prototype.flush=function(r){this._active=!0;var o=this._scheduled;this._scheduled=void 0;var n=this.actions,i;r=r||n.shift();do if(i=r.execute(r.state,r.delay))break;while((r=n[0])&&r.id===o&&n.shift());if(this._active=!1,i){for(;(r=n[0])&&r.id===o&&n.shift();)r.unsubscribe();throw i}},t}(yt);var me=new _o(Mo);var S=new j(function(e){return e.complete()});function Yt(e){return e&&H(e.schedule)}function Hr(e){return e[e.length-1]}function Xe(e){return H(Hr(e))?e.pop():void 0}function ke(e){return Yt(Hr(e))?e.pop():void 0}function Bt(e,t){return typeof Hr(e)=="number"?e.pop():t}var xt=function(e){return e&&typeof e.length=="number"&&typeof e!="function"};function Gt(e){return H(e==null?void 0:e.then)}function Jt(e){return H(e[bt])}function Xt(e){return Symbol.asyncIterator&&H(e==null?void 0:e[Symbol.asyncIterator])}function Zt(e){return new TypeError("You provided "+(e!==null&&typeof e=="object"?"an invalid object":"'"+e+"'")+" where a stream was expected. You can provide an Observable, Promise, ReadableStream, Array, AsyncIterable, or Iterable.")}function Zi(){return typeof Symbol!="function"||!Symbol.iterator?"@@iterator":Symbol.iterator}var er=Zi();function tr(e){return H(e==null?void 0:e[er])}function rr(e){return fo(this,arguments,function(){var r,o,n,i;return Nt(this,function(a){switch(a.label){case 0:r=e.getReader(),a.label=1;case 1:a.trys.push([1,,9,10]),a.label=2;case 2:return[4,nt(r.read())];case 3:return o=a.sent(),n=o.value,i=o.done,i?[4,nt(void 0)]:[3,5];case 4:return[2,a.sent()];case 5:return[4,nt(n)];case 6:return[4,a.sent()];case 7:return a.sent(),[3,2];case 8:return[3,10];case 9:return r.releaseLock(),[7];case 10:return[2]}})})}function or(e){return H(e==null?void 0:e.getReader)}function U(e){if(e instanceof j)return e;if(e!=null){if(Jt(e))return ea(e);if(xt(e))return ta(e);if(Gt(e))return ra(e);if(Xt(e))return Ao(e);if(tr(e))return oa(e);if(or(e))return na(e)}throw Zt(e)}function ea(e){return new j(function(t){var r=e[bt]();if(H(r.subscribe))return r.subscribe(t);throw new TypeError("Provided object does not correctly implement Symbol.observable")})}function ta(e){return new j(function(t){for(var r=0;r=2;return function(o){return o.pipe(e?b(function(n,i){return e(n,i,o)}):le,Te(1),r?De(t):Qo(function(){return new ir}))}}function jr(e){return e<=0?function(){return S}:E(function(t,r){var o=[];t.subscribe(T(r,function(n){o.push(n),e=2,!0))}function pe(e){e===void 0&&(e={});var t=e.connector,r=t===void 0?function(){return new g}:t,o=e.resetOnError,n=o===void 0?!0:o,i=e.resetOnComplete,a=i===void 0?!0:i,s=e.resetOnRefCountZero,p=s===void 0?!0:s;return function(c){var l,f,u,d=0,y=!1,L=!1,X=function(){f==null||f.unsubscribe(),f=void 0},te=function(){X(),l=u=void 0,y=L=!1},J=function(){var k=l;te(),k==null||k.unsubscribe()};return E(function(k,ft){d++,!L&&!y&&X();var qe=u=u!=null?u:r();ft.add(function(){d--,d===0&&!L&&!y&&(f=Ur(J,p))}),qe.subscribe(ft),!l&&d>0&&(l=new at({next:function(Fe){return qe.next(Fe)},error:function(Fe){L=!0,X(),f=Ur(te,n,Fe),qe.error(Fe)},complete:function(){y=!0,X(),f=Ur(te,a),qe.complete()}}),U(k).subscribe(l))})(c)}}function Ur(e,t){for(var r=[],o=2;oe.next(document)),e}function P(e,t=document){return Array.from(t.querySelectorAll(e))}function R(e,t=document){let r=fe(e,t);if(typeof r=="undefined")throw new ReferenceError(`Missing element: expected "${e}" to be present`);return r}function fe(e,t=document){return t.querySelector(e)||void 0}function Ie(){var e,t,r,o;return(o=(r=(t=(e=document.activeElement)==null?void 0:e.shadowRoot)==null?void 0:t.activeElement)!=null?r:document.activeElement)!=null?o:void 0}var wa=O(h(document.body,"focusin"),h(document.body,"focusout")).pipe(_e(1),Q(void 0),m(()=>Ie()||document.body),G(1));function et(e){return wa.pipe(m(t=>e.contains(t)),K())}function $t(e,t){return C(()=>O(h(e,"mouseenter").pipe(m(()=>!0)),h(e,"mouseleave").pipe(m(()=>!1))).pipe(t?Ht(r=>Le(+!r*t)):le,Q(e.matches(":hover"))))}function Jo(e,t){if(typeof t=="string"||typeof t=="number")e.innerHTML+=t.toString();else if(t instanceof Node)e.appendChild(t);else if(Array.isArray(t))for(let r of t)Jo(e,r)}function x(e,t,...r){let o=document.createElement(e);if(t)for(let n of Object.keys(t))typeof t[n]!="undefined"&&(typeof t[n]!="boolean"?o.setAttribute(n,t[n]):o.setAttribute(n,""));for(let n of r)Jo(o,n);return o}function sr(e){if(e>999){let t=+((e-950)%1e3>99);return`${((e+1e-6)/1e3).toFixed(t)}k`}else return e.toString()}function Tt(e){let t=x("script",{src:e});return C(()=>(document.head.appendChild(t),O(h(t,"load"),h(t,"error").pipe(v(()=>$r(()=>new ReferenceError(`Invalid script: ${e}`))))).pipe(m(()=>{}),_(()=>document.head.removeChild(t)),Te(1))))}var Xo=new g,Ta=C(()=>typeof ResizeObserver=="undefined"?Tt("https://unpkg.com/resize-observer-polyfill"):I(void 0)).pipe(m(()=>new ResizeObserver(e=>e.forEach(t=>Xo.next(t)))),v(e=>O(Ye,I(e)).pipe(_(()=>e.disconnect()))),G(1));function ce(e){return{width:e.offsetWidth,height:e.offsetHeight}}function ge(e){let t=e;for(;t.clientWidth===0&&t.parentElement;)t=t.parentElement;return Ta.pipe(w(r=>r.observe(t)),v(r=>Xo.pipe(b(o=>o.target===t),_(()=>r.unobserve(t)))),m(()=>ce(e)),Q(ce(e)))}function St(e){return{width:e.scrollWidth,height:e.scrollHeight}}function cr(e){let t=e.parentElement;for(;t&&(e.scrollWidth<=t.scrollWidth&&e.scrollHeight<=t.scrollHeight);)t=(e=t).parentElement;return t?e:void 0}function Zo(e){let t=[],r=e.parentElement;for(;r;)(e.clientWidth>r.clientWidth||e.clientHeight>r.clientHeight)&&t.push(r),r=(e=r).parentElement;return t.length===0&&t.push(document.documentElement),t}function Ve(e){return{x:e.offsetLeft,y:e.offsetTop}}function en(e){let t=e.getBoundingClientRect();return{x:t.x+window.scrollX,y:t.y+window.scrollY}}function tn(e){return O(h(window,"load"),h(window,"resize")).pipe(Me(0,me),m(()=>Ve(e)),Q(Ve(e)))}function pr(e){return{x:e.scrollLeft,y:e.scrollTop}}function Ne(e){return O(h(e,"scroll"),h(window,"scroll"),h(window,"resize")).pipe(Me(0,me),m(()=>pr(e)),Q(pr(e)))}var rn=new g,Sa=C(()=>I(new IntersectionObserver(e=>{for(let t of e)rn.next(t)},{threshold:0}))).pipe(v(e=>O(Ye,I(e)).pipe(_(()=>e.disconnect()))),G(1));function tt(e){return Sa.pipe(w(t=>t.observe(e)),v(t=>rn.pipe(b(({target:r})=>r===e),_(()=>t.unobserve(e)),m(({isIntersecting:r})=>r))))}function on(e,t=16){return Ne(e).pipe(m(({y:r})=>{let o=ce(e),n=St(e);return r>=n.height-o.height-t}),K())}var lr={drawer:R("[data-md-toggle=drawer]"),search:R("[data-md-toggle=search]")};function nn(e){return lr[e].checked}function Je(e,t){lr[e].checked!==t&&lr[e].click()}function ze(e){let t=lr[e];return h(t,"change").pipe(m(()=>t.checked),Q(t.checked))}function Oa(e,t){switch(e.constructor){case HTMLInputElement:return e.type==="radio"?/^Arrow/.test(t):!0;case HTMLSelectElement:case HTMLTextAreaElement:return!0;default:return e.isContentEditable}}function La(){return O(h(window,"compositionstart").pipe(m(()=>!0)),h(window,"compositionend").pipe(m(()=>!1))).pipe(Q(!1))}function an(){let e=h(window,"keydown").pipe(b(t=>!(t.metaKey||t.ctrlKey)),m(t=>({mode:nn("search")?"search":"global",type:t.key,claim(){t.preventDefault(),t.stopPropagation()}})),b(({mode:t,type:r})=>{if(t==="global"){let o=Ie();if(typeof o!="undefined")return!Oa(o,r)}return!0}),pe());return La().pipe(v(t=>t?S:e))}function ye(){return new URL(location.href)}function lt(e,t=!1){if(B("navigation.instant")&&!t){let r=x("a",{href:e.href});document.body.appendChild(r),r.click(),r.remove()}else location.href=e.href}function sn(){return new g}function cn(){return location.hash.slice(1)}function pn(e){let t=x("a",{href:e});t.addEventListener("click",r=>r.stopPropagation()),t.click()}function Ma(e){return O(h(window,"hashchange"),e).pipe(m(cn),Q(cn()),b(t=>t.length>0),G(1))}function ln(e){return Ma(e).pipe(m(t=>fe(`[id="${t}"]`)),b(t=>typeof t!="undefined"))}function Pt(e){let t=matchMedia(e);return ar(r=>t.addListener(()=>r(t.matches))).pipe(Q(t.matches))}function mn(){let e=matchMedia("print");return O(h(window,"beforeprint").pipe(m(()=>!0)),h(window,"afterprint").pipe(m(()=>!1))).pipe(Q(e.matches))}function Nr(e,t){return e.pipe(v(r=>r?t():S))}function zr(e,t){return new j(r=>{let o=new XMLHttpRequest;return o.open("GET",`${e}`),o.responseType="blob",o.addEventListener("load",()=>{o.status>=200&&o.status<300?(r.next(o.response),r.complete()):r.error(new Error(o.statusText))}),o.addEventListener("error",()=>{r.error(new Error("Network error"))}),o.addEventListener("abort",()=>{r.complete()}),typeof(t==null?void 0:t.progress$)!="undefined"&&(o.addEventListener("progress",n=>{var i;if(n.lengthComputable)t.progress$.next(n.loaded/n.total*100);else{let a=(i=o.getResponseHeader("Content-Length"))!=null?i:0;t.progress$.next(n.loaded/+a*100)}}),t.progress$.next(5)),o.send(),()=>o.abort()})}function je(e,t){return zr(e,t).pipe(v(r=>r.text()),m(r=>JSON.parse(r)),G(1))}function fn(e,t){let r=new DOMParser;return zr(e,t).pipe(v(o=>o.text()),m(o=>r.parseFromString(o,"text/html")),G(1))}function un(e,t){let r=new DOMParser;return zr(e,t).pipe(v(o=>o.text()),m(o=>r.parseFromString(o,"text/xml")),G(1))}function dn(){return{x:Math.max(0,scrollX),y:Math.max(0,scrollY)}}function hn(){return O(h(window,"scroll",{passive:!0}),h(window,"resize",{passive:!0})).pipe(m(dn),Q(dn()))}function bn(){return{width:innerWidth,height:innerHeight}}function vn(){return h(window,"resize",{passive:!0}).pipe(m(bn),Q(bn()))}function gn(){return z([hn(),vn()]).pipe(m(([e,t])=>({offset:e,size:t})),G(1))}function mr(e,{viewport$:t,header$:r}){let o=t.pipe(ee("size")),n=z([o,r]).pipe(m(()=>Ve(e)));return z([r,t,n]).pipe(m(([{height:i},{offset:a,size:s},{x:p,y:c}])=>({offset:{x:a.x-p,y:a.y-c+i},size:s})))}function _a(e){return h(e,"message",t=>t.data)}function Aa(e){let t=new g;return t.subscribe(r=>e.postMessage(r)),t}function yn(e,t=new Worker(e)){let r=_a(t),o=Aa(t),n=new g;n.subscribe(o);let i=o.pipe(Z(),ie(!0));return n.pipe(Z(),Re(r.pipe(W(i))),pe())}var Ca=R("#__config"),Ot=JSON.parse(Ca.textContent);Ot.base=`${new URL(Ot.base,ye())}`;function xe(){return Ot}function B(e){return Ot.features.includes(e)}function Ee(e,t){return typeof t!="undefined"?Ot.translations[e].replace("#",t.toString()):Ot.translations[e]}function Se(e,t=document){return R(`[data-md-component=${e}]`,t)}function ae(e,t=document){return P(`[data-md-component=${e}]`,t)}function ka(e){let t=R(".md-typeset > :first-child",e);return h(t,"click",{once:!0}).pipe(m(()=>R(".md-typeset",e)),m(r=>({hash:__md_hash(r.innerHTML)})))}function xn(e){if(!B("announce.dismiss")||!e.childElementCount)return S;if(!e.hidden){let t=R(".md-typeset",e);__md_hash(t.innerHTML)===__md_get("__announce")&&(e.hidden=!0)}return C(()=>{let t=new g;return t.subscribe(({hash:r})=>{e.hidden=!0,__md_set("__announce",r)}),ka(e).pipe(w(r=>t.next(r)),_(()=>t.complete()),m(r=>$({ref:e},r)))})}function Ha(e,{target$:t}){return t.pipe(m(r=>({hidden:r!==e})))}function En(e,t){let r=new g;return r.subscribe(({hidden:o})=>{e.hidden=o}),Ha(e,t).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))}function Rt(e,t){return t==="inline"?x("div",{class:"md-tooltip md-tooltip--inline",id:e,role:"tooltip"},x("div",{class:"md-tooltip__inner md-typeset"})):x("div",{class:"md-tooltip",id:e,role:"tooltip"},x("div",{class:"md-tooltip__inner md-typeset"}))}function wn(...e){return x("div",{class:"md-tooltip2",role:"tooltip"},x("div",{class:"md-tooltip2__inner md-typeset"},e))}function Tn(e,t){if(t=t?`${t}_annotation_${e}`:void 0,t){let r=t?`#${t}`:void 0;return x("aside",{class:"md-annotation",tabIndex:0},Rt(t),x("a",{href:r,class:"md-annotation__index",tabIndex:-1},x("span",{"data-md-annotation-id":e})))}else return x("aside",{class:"md-annotation",tabIndex:0},Rt(t),x("span",{class:"md-annotation__index",tabIndex:-1},x("span",{"data-md-annotation-id":e})))}function Sn(e){return x("button",{class:"md-clipboard md-icon",title:Ee("clipboard.copy"),"data-clipboard-target":`#${e} > code`})}var Ln=Mt(qr());function Qr(e,t){let r=t&2,o=t&1,n=Object.keys(e.terms).filter(p=>!e.terms[p]).reduce((p,c)=>[...p,x("del",null,(0,Ln.default)(c))," "],[]).slice(0,-1),i=xe(),a=new URL(e.location,i.base);B("search.highlight")&&a.searchParams.set("h",Object.entries(e.terms).filter(([,p])=>p).reduce((p,[c])=>`${p} ${c}`.trim(),""));let{tags:s}=xe();return x("a",{href:`${a}`,class:"md-search-result__link",tabIndex:-1},x("article",{class:"md-search-result__article md-typeset","data-md-score":e.score.toFixed(2)},r>0&&x("div",{class:"md-search-result__icon md-icon"}),r>0&&x("h1",null,e.title),r<=0&&x("h2",null,e.title),o>0&&e.text.length>0&&e.text,e.tags&&x("nav",{class:"md-tags"},e.tags.map(p=>{let c=s?p in s?`md-tag-icon md-tag--${s[p]}`:"md-tag-icon":"";return x("span",{class:`md-tag ${c}`},p)})),o>0&&n.length>0&&x("p",{class:"md-search-result__terms"},Ee("search.result.term.missing"),": ",...n)))}function Mn(e){let t=e[0].score,r=[...e],o=xe(),n=r.findIndex(l=>!`${new URL(l.location,o.base)}`.includes("#")),[i]=r.splice(n,1),a=r.findIndex(l=>l.scoreQr(l,1)),...p.length?[x("details",{class:"md-search-result__more"},x("summary",{tabIndex:-1},x("div",null,p.length>0&&p.length===1?Ee("search.result.more.one"):Ee("search.result.more.other",p.length))),...p.map(l=>Qr(l,1)))]:[]];return x("li",{class:"md-search-result__item"},c)}function _n(e){return x("ul",{class:"md-source__facts"},Object.entries(e).map(([t,r])=>x("li",{class:`md-source__fact md-source__fact--${t}`},typeof r=="number"?sr(r):r)))}function Kr(e){let t=`tabbed-control tabbed-control--${e}`;return x("div",{class:t,hidden:!0},x("button",{class:"tabbed-button",tabIndex:-1,"aria-hidden":"true"}))}function An(e){return x("div",{class:"md-typeset__scrollwrap"},x("div",{class:"md-typeset__table"},e))}function Ra(e){var o;let t=xe(),r=new URL(`../${e.version}/`,t.base);return x("li",{class:"md-version__item"},x("a",{href:`${r}`,class:"md-version__link"},e.title,((o=t.version)==null?void 0:o.alias)&&e.aliases.length>0&&x("span",{class:"md-version__alias"},e.aliases[0])))}function Cn(e,t){var o;let r=xe();return e=e.filter(n=>{var i;return!((i=n.properties)!=null&&i.hidden)}),x("div",{class:"md-version"},x("button",{class:"md-version__current","aria-label":Ee("select.version")},t.title,((o=r.version)==null?void 0:o.alias)&&t.aliases.length>0&&x("span",{class:"md-version__alias"},t.aliases[0])),x("ul",{class:"md-version__list"},e.map(Ra)))}var Ia=0;function ja(e){let t=z([et(e),$t(e)]).pipe(m(([o,n])=>o||n),K()),r=C(()=>Zo(e)).pipe(ne(Ne),pt(1),He(t),m(()=>en(e)));return t.pipe(Ae(o=>o),v(()=>z([t,r])),m(([o,n])=>({active:o,offset:n})),pe())}function Fa(e,t){let{content$:r,viewport$:o}=t,n=`__tooltip2_${Ia++}`;return C(()=>{let i=new g,a=new _r(!1);i.pipe(Z(),ie(!1)).subscribe(a);let s=a.pipe(Ht(c=>Le(+!c*250,kr)),K(),v(c=>c?r:S),w(c=>c.id=n),pe());z([i.pipe(m(({active:c})=>c)),s.pipe(v(c=>$t(c,250)),Q(!1))]).pipe(m(c=>c.some(l=>l))).subscribe(a);let p=a.pipe(b(c=>c),re(s,o),m(([c,l,{size:f}])=>{let u=e.getBoundingClientRect(),d=u.width/2;if(l.role==="tooltip")return{x:d,y:8+u.height};if(u.y>=f.height/2){let{height:y}=ce(l);return{x:d,y:-16-y}}else return{x:d,y:16+u.height}}));return z([s,i,p]).subscribe(([c,{offset:l},f])=>{c.style.setProperty("--md-tooltip-host-x",`${l.x}px`),c.style.setProperty("--md-tooltip-host-y",`${l.y}px`),c.style.setProperty("--md-tooltip-x",`${f.x}px`),c.style.setProperty("--md-tooltip-y",`${f.y}px`),c.classList.toggle("md-tooltip2--top",f.y<0),c.classList.toggle("md-tooltip2--bottom",f.y>=0)}),a.pipe(b(c=>c),re(s,(c,l)=>l),b(c=>c.role==="tooltip")).subscribe(c=>{let l=ce(R(":scope > *",c));c.style.setProperty("--md-tooltip-width",`${l.width}px`),c.style.setProperty("--md-tooltip-tail","0px")}),a.pipe(K(),ve(me),re(s)).subscribe(([c,l])=>{l.classList.toggle("md-tooltip2--active",c)}),z([a.pipe(b(c=>c)),s]).subscribe(([c,l])=>{l.role==="dialog"?(e.setAttribute("aria-controls",n),e.setAttribute("aria-haspopup","dialog")):e.setAttribute("aria-describedby",n)}),a.pipe(b(c=>!c)).subscribe(()=>{e.removeAttribute("aria-controls"),e.removeAttribute("aria-describedby"),e.removeAttribute("aria-haspopup")}),ja(e).pipe(w(c=>i.next(c)),_(()=>i.complete()),m(c=>$({ref:e},c)))})}function mt(e,{viewport$:t},r=document.body){return Fa(e,{content$:new j(o=>{let n=e.title,i=wn(n);return o.next(i),e.removeAttribute("title"),r.append(i),()=>{i.remove(),e.setAttribute("title",n)}}),viewport$:t})}function Ua(e,t){let r=C(()=>z([tn(e),Ne(t)])).pipe(m(([{x:o,y:n},i])=>{let{width:a,height:s}=ce(e);return{x:o-i.x+a/2,y:n-i.y+s/2}}));return et(e).pipe(v(o=>r.pipe(m(n=>({active:o,offset:n})),Te(+!o||1/0))))}function kn(e,t,{target$:r}){let[o,n]=Array.from(e.children);return C(()=>{let i=new g,a=i.pipe(Z(),ie(!0));return i.subscribe({next({offset:s}){e.style.setProperty("--md-tooltip-x",`${s.x}px`),e.style.setProperty("--md-tooltip-y",`${s.y}px`)},complete(){e.style.removeProperty("--md-tooltip-x"),e.style.removeProperty("--md-tooltip-y")}}),tt(e).pipe(W(a)).subscribe(s=>{e.toggleAttribute("data-md-visible",s)}),O(i.pipe(b(({active:s})=>s)),i.pipe(_e(250),b(({active:s})=>!s))).subscribe({next({active:s}){s?e.prepend(o):o.remove()},complete(){e.prepend(o)}}),i.pipe(Me(16,me)).subscribe(({active:s})=>{o.classList.toggle("md-tooltip--active",s)}),i.pipe(pt(125,me),b(()=>!!e.offsetParent),m(()=>e.offsetParent.getBoundingClientRect()),m(({x:s})=>s)).subscribe({next(s){s?e.style.setProperty("--md-tooltip-0",`${-s}px`):e.style.removeProperty("--md-tooltip-0")},complete(){e.style.removeProperty("--md-tooltip-0")}}),h(n,"click").pipe(W(a),b(s=>!(s.metaKey||s.ctrlKey))).subscribe(s=>{s.stopPropagation(),s.preventDefault()}),h(n,"mousedown").pipe(W(a),re(i)).subscribe(([s,{active:p}])=>{var c;if(s.button!==0||s.metaKey||s.ctrlKey)s.preventDefault();else if(p){s.preventDefault();let l=e.parentElement.closest(".md-annotation");l instanceof HTMLElement?l.focus():(c=Ie())==null||c.blur()}}),r.pipe(W(a),b(s=>s===o),Ge(125)).subscribe(()=>e.focus()),Ua(e,t).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))})}function Wa(e){return e.tagName==="CODE"?P(".c, .c1, .cm",e):[e]}function Da(e){let t=[];for(let r of Wa(e)){let o=[],n=document.createNodeIterator(r,NodeFilter.SHOW_TEXT);for(let i=n.nextNode();i;i=n.nextNode())o.push(i);for(let i of o){let a;for(;a=/(\(\d+\))(!)?/.exec(i.textContent);){let[,s,p]=a;if(typeof p=="undefined"){let c=i.splitText(a.index);i=c.splitText(s.length),t.push(c)}else{i.textContent=s,t.push(i);break}}}}return t}function Hn(e,t){t.append(...Array.from(e.childNodes))}function fr(e,t,{target$:r,print$:o}){let n=t.closest("[id]"),i=n==null?void 0:n.id,a=new Map;for(let s of Da(t)){let[,p]=s.textContent.match(/\((\d+)\)/);fe(`:scope > li:nth-child(${p})`,e)&&(a.set(p,Tn(p,i)),s.replaceWith(a.get(p)))}return a.size===0?S:C(()=>{let s=new g,p=s.pipe(Z(),ie(!0)),c=[];for(let[l,f]of a)c.push([R(".md-typeset",f),R(`:scope > li:nth-child(${l})`,e)]);return o.pipe(W(p)).subscribe(l=>{e.hidden=!l,e.classList.toggle("md-annotation-list",l);for(let[f,u]of c)l?Hn(f,u):Hn(u,f)}),O(...[...a].map(([,l])=>kn(l,t,{target$:r}))).pipe(_(()=>s.complete()),pe())})}function $n(e){if(e.nextElementSibling){let t=e.nextElementSibling;if(t.tagName==="OL")return t;if(t.tagName==="P"&&!t.children.length)return $n(t)}}function Pn(e,t){return C(()=>{let r=$n(e);return typeof r!="undefined"?fr(r,e,t):S})}var Rn=Mt(Br());var Va=0;function In(e){if(e.nextElementSibling){let t=e.nextElementSibling;if(t.tagName==="OL")return t;if(t.tagName==="P"&&!t.children.length)return In(t)}}function Na(e){return ge(e).pipe(m(({width:t})=>({scrollable:St(e).width>t})),ee("scrollable"))}function jn(e,t){let{matches:r}=matchMedia("(hover)"),o=C(()=>{let n=new g,i=n.pipe(jr(1));n.subscribe(({scrollable:c})=>{c&&r?e.setAttribute("tabindex","0"):e.removeAttribute("tabindex")});let a=[];if(Rn.default.isSupported()&&(e.closest(".copy")||B("content.code.copy")&&!e.closest(".no-copy"))){let c=e.closest("pre");c.id=`__code_${Va++}`;let l=Sn(c.id);c.insertBefore(l,e),B("content.tooltips")&&a.push(mt(l,{viewport$}))}let s=e.closest(".highlight");if(s instanceof HTMLElement){let c=In(s);if(typeof c!="undefined"&&(s.classList.contains("annotate")||B("content.code.annotate"))){let l=fr(c,e,t);a.push(ge(s).pipe(W(i),m(({width:f,height:u})=>f&&u),K(),v(f=>f?l:S)))}}return P(":scope > span[id]",e).length&&e.classList.add("md-code__content"),Na(e).pipe(w(c=>n.next(c)),_(()=>n.complete()),m(c=>$({ref:e},c)),Re(...a))});return B("content.lazy")?tt(e).pipe(b(n=>n),Te(1),v(()=>o)):o}function za(e,{target$:t,print$:r}){let o=!0;return O(t.pipe(m(n=>n.closest("details:not([open])")),b(n=>e===n),m(()=>({action:"open",reveal:!0}))),r.pipe(b(n=>n||!o),w(()=>o=e.open),m(n=>({action:n?"open":"close"}))))}function Fn(e,t){return C(()=>{let r=new g;return r.subscribe(({action:o,reveal:n})=>{e.toggleAttribute("open",o==="open"),n&&e.scrollIntoView()}),za(e,t).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}var Un=".node circle,.node ellipse,.node path,.node polygon,.node rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}marker{fill:var(--md-mermaid-edge-color)!important}.edgeLabel .label rect{fill:#0000}.flowchartTitleText{fill:var(--md-mermaid-label-fg-color)}.label{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.label foreignObject{line-height:normal;overflow:visible}.label div .edgeLabel{color:var(--md-mermaid-label-fg-color)}.edgeLabel,.edgeLabel p,.label div .edgeLabel{background-color:var(--md-mermaid-label-bg-color)}.edgeLabel,.edgeLabel p{fill:var(--md-mermaid-label-bg-color);color:var(--md-mermaid-edge-color)}.edgePath .path,.flowchart-link{stroke:var(--md-mermaid-edge-color);stroke-width:.05rem}.edgePath .arrowheadPath{fill:var(--md-mermaid-edge-color);stroke:none}.cluster rect{fill:var(--md-default-fg-color--lightest);stroke:var(--md-default-fg-color--lighter)}.cluster span{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}g #flowchart-circleEnd,g #flowchart-circleStart,g #flowchart-crossEnd,g #flowchart-crossStart,g #flowchart-pointEnd,g #flowchart-pointStart{stroke:none}.classDiagramTitleText{fill:var(--md-mermaid-label-fg-color)}g.classGroup line,g.classGroup rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}g.classGroup text{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.classLabel .box{fill:var(--md-mermaid-label-bg-color);background-color:var(--md-mermaid-label-bg-color);opacity:1}.classLabel .label{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.node .divider{stroke:var(--md-mermaid-node-fg-color)}.relation{stroke:var(--md-mermaid-edge-color)}.cardinality{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.cardinality text{fill:inherit!important}defs #classDiagram-compositionEnd,defs #classDiagram-compositionStart,defs #classDiagram-dependencyEnd,defs #classDiagram-dependencyStart,defs #classDiagram-extensionEnd,defs #classDiagram-extensionStart{fill:var(--md-mermaid-edge-color)!important;stroke:var(--md-mermaid-edge-color)!important}defs #classDiagram-aggregationEnd,defs #classDiagram-aggregationStart{fill:var(--md-mermaid-label-bg-color)!important;stroke:var(--md-mermaid-edge-color)!important}.statediagramTitleText{fill:var(--md-mermaid-label-fg-color)}g.stateGroup rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}g.stateGroup .state-title{fill:var(--md-mermaid-label-fg-color)!important;font-family:var(--md-mermaid-font-family)}g.stateGroup .composit{fill:var(--md-mermaid-label-bg-color)}.nodeLabel,.nodeLabel p{color:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}a .nodeLabel{text-decoration:underline}.node circle.state-end,.node circle.state-start,.start-state{fill:var(--md-mermaid-edge-color);stroke:none}.end-state-inner,.end-state-outer{fill:var(--md-mermaid-edge-color)}.end-state-inner,.node circle.state-end{stroke:var(--md-mermaid-label-bg-color)}.transition{stroke:var(--md-mermaid-edge-color)}[id^=state-fork] rect,[id^=state-join] rect{fill:var(--md-mermaid-edge-color)!important;stroke:none!important}.statediagram-cluster.statediagram-cluster .inner{fill:var(--md-default-bg-color)}.statediagram-cluster rect{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}.statediagram-state rect.divider{fill:var(--md-default-fg-color--lightest);stroke:var(--md-default-fg-color--lighter)}defs #statediagram-barbEnd{stroke:var(--md-mermaid-edge-color)}.entityTitleText{fill:var(--md-mermaid-label-fg-color)}.attributeBoxEven,.attributeBoxOdd{fill:var(--md-mermaid-node-bg-color);stroke:var(--md-mermaid-node-fg-color)}.entityBox{fill:var(--md-mermaid-label-bg-color);stroke:var(--md-mermaid-node-fg-color)}.entityLabel{fill:var(--md-mermaid-label-fg-color);font-family:var(--md-mermaid-font-family)}.relationshipLabelBox{fill:var(--md-mermaid-label-bg-color);fill-opacity:1;background-color:var(--md-mermaid-label-bg-color);opacity:1}.relationshipLabel{fill:var(--md-mermaid-label-fg-color)}.relationshipLine{stroke:var(--md-mermaid-edge-color)}defs #ONE_OR_MORE_END *,defs #ONE_OR_MORE_START *,defs #ONLY_ONE_END *,defs #ONLY_ONE_START *,defs #ZERO_OR_MORE_END *,defs #ZERO_OR_MORE_START *,defs #ZERO_OR_ONE_END *,defs #ZERO_OR_ONE_START *{stroke:var(--md-mermaid-edge-color)!important}defs #ZERO_OR_MORE_END circle,defs #ZERO_OR_MORE_START circle{fill:var(--md-mermaid-label-bg-color)}text:not([class]):last-child{fill:var(--md-mermaid-label-fg-color)}.actor{fill:var(--md-mermaid-sequence-actor-bg-color);stroke:var(--md-mermaid-sequence-actor-border-color)}text.actor>tspan{fill:var(--md-mermaid-sequence-actor-fg-color);font-family:var(--md-mermaid-font-family)}line{stroke:var(--md-mermaid-sequence-actor-line-color)}.actor-man circle,.actor-man line{fill:var(--md-mermaid-sequence-actorman-bg-color);stroke:var(--md-mermaid-sequence-actorman-line-color)}.messageLine0,.messageLine1{stroke:var(--md-mermaid-sequence-message-line-color)}.note{fill:var(--md-mermaid-sequence-note-bg-color);stroke:var(--md-mermaid-sequence-note-border-color)}.loopText,.loopText>tspan,.messageText,.noteText>tspan{stroke:none;font-family:var(--md-mermaid-font-family)!important}.messageText{fill:var(--md-mermaid-sequence-message-fg-color)}.loopText,.loopText>tspan{fill:var(--md-mermaid-sequence-loop-fg-color)}.noteText>tspan{fill:var(--md-mermaid-sequence-note-fg-color)}#arrowhead path{fill:var(--md-mermaid-sequence-message-line-color);stroke:none}.loopLine{fill:var(--md-mermaid-sequence-loop-bg-color);stroke:var(--md-mermaid-sequence-loop-border-color)}.labelBox{fill:var(--md-mermaid-sequence-label-bg-color);stroke:none}.labelText,.labelText>span{fill:var(--md-mermaid-sequence-label-fg-color);font-family:var(--md-mermaid-font-family)}.sequenceNumber{fill:var(--md-mermaid-sequence-number-fg-color)}rect.rect{fill:var(--md-mermaid-sequence-box-bg-color);stroke:none}rect.rect+text.text{fill:var(--md-mermaid-sequence-box-fg-color)}defs #sequencenumber{fill:var(--md-mermaid-sequence-number-bg-color)!important}";var Gr,Qa=0;function Ka(){return typeof mermaid=="undefined"||mermaid instanceof Element?Tt("https://unpkg.com/mermaid@11/dist/mermaid.min.js"):I(void 0)}function Wn(e){return e.classList.remove("mermaid"),Gr||(Gr=Ka().pipe(w(()=>mermaid.initialize({startOnLoad:!1,themeCSS:Un,sequence:{actorFontSize:"16px",messageFontSize:"16px",noteFontSize:"16px"}})),m(()=>{}),G(1))),Gr.subscribe(()=>co(this,null,function*(){e.classList.add("mermaid");let t=`__mermaid_${Qa++}`,r=x("div",{class:"mermaid"}),o=e.textContent,{svg:n,fn:i}=yield mermaid.render(t,o),a=r.attachShadow({mode:"closed"});a.innerHTML=n,e.replaceWith(r),i==null||i(a)})),Gr.pipe(m(()=>({ref:e})))}var Dn=x("table");function Vn(e){return e.replaceWith(Dn),Dn.replaceWith(An(e)),I({ref:e})}function Ya(e){let t=e.find(r=>r.checked)||e[0];return O(...e.map(r=>h(r,"change").pipe(m(()=>R(`label[for="${r.id}"]`))))).pipe(Q(R(`label[for="${t.id}"]`)),m(r=>({active:r})))}function Nn(e,{viewport$:t,target$:r}){let o=R(".tabbed-labels",e),n=P(":scope > input",e),i=Kr("prev");e.append(i);let a=Kr("next");return e.append(a),C(()=>{let s=new g,p=s.pipe(Z(),ie(!0));z([s,ge(e),tt(e)]).pipe(W(p),Me(1,me)).subscribe({next([{active:c},l]){let f=Ve(c),{width:u}=ce(c);e.style.setProperty("--md-indicator-x",`${f.x}px`),e.style.setProperty("--md-indicator-width",`${u}px`);let d=pr(o);(f.xd.x+l.width)&&o.scrollTo({left:Math.max(0,f.x-16),behavior:"smooth"})},complete(){e.style.removeProperty("--md-indicator-x"),e.style.removeProperty("--md-indicator-width")}}),z([Ne(o),ge(o)]).pipe(W(p)).subscribe(([c,l])=>{let f=St(o);i.hidden=c.x<16,a.hidden=c.x>f.width-l.width-16}),O(h(i,"click").pipe(m(()=>-1)),h(a,"click").pipe(m(()=>1))).pipe(W(p)).subscribe(c=>{let{width:l}=ce(o);o.scrollBy({left:l*c,behavior:"smooth"})}),r.pipe(W(p),b(c=>n.includes(c))).subscribe(c=>c.click()),o.classList.add("tabbed-labels--linked");for(let c of n){let l=R(`label[for="${c.id}"]`);l.replaceChildren(x("a",{href:`#${l.htmlFor}`,tabIndex:-1},...Array.from(l.childNodes))),h(l.firstElementChild,"click").pipe(W(p),b(f=>!(f.metaKey||f.ctrlKey)),w(f=>{f.preventDefault(),f.stopPropagation()})).subscribe(()=>{history.replaceState({},"",`#${l.htmlFor}`),l.click()})}return B("content.tabs.link")&&s.pipe(Ce(1),re(t)).subscribe(([{active:c},{offset:l}])=>{let f=c.innerText.trim();if(c.hasAttribute("data-md-switching"))c.removeAttribute("data-md-switching");else{let u=e.offsetTop-l.y;for(let y of P("[data-tabs]"))for(let L of P(":scope > input",y)){let X=R(`label[for="${L.id}"]`);if(X!==c&&X.innerText.trim()===f){X.setAttribute("data-md-switching",""),L.click();break}}window.scrollTo({top:e.offsetTop-u});let d=__md_get("__tabs")||[];__md_set("__tabs",[...new Set([f,...d])])}}),s.pipe(W(p)).subscribe(()=>{for(let c of P("audio, video",e))c.pause()}),Ya(n).pipe(w(c=>s.next(c)),_(()=>s.complete()),m(c=>$({ref:e},c)))}).pipe(Ke(se))}function zn(e,{viewport$:t,target$:r,print$:o}){return O(...P(".annotate:not(.highlight)",e).map(n=>Pn(n,{target$:r,print$:o})),...P("pre:not(.mermaid) > code",e).map(n=>jn(n,{target$:r,print$:o})),...P("pre.mermaid",e).map(n=>Wn(n)),...P("table:not([class])",e).map(n=>Vn(n)),...P("details",e).map(n=>Fn(n,{target$:r,print$:o})),...P("[data-tabs]",e).map(n=>Nn(n,{viewport$:t,target$:r})),...P("[title]",e).filter(()=>B("content.tooltips")).map(n=>mt(n,{viewport$:t})))}function Ba(e,{alert$:t}){return t.pipe(v(r=>O(I(!0),I(!1).pipe(Ge(2e3))).pipe(m(o=>({message:r,active:o})))))}function qn(e,t){let r=R(".md-typeset",e);return C(()=>{let o=new g;return o.subscribe(({message:n,active:i})=>{e.classList.toggle("md-dialog--active",i),r.textContent=n}),Ba(e,t).pipe(w(n=>o.next(n)),_(()=>o.complete()),m(n=>$({ref:e},n)))})}var Ga=0;function Ja(e,t){document.body.append(e);let{width:r}=ce(e);e.style.setProperty("--md-tooltip-width",`${r}px`),e.remove();let o=cr(t),n=typeof o!="undefined"?Ne(o):I({x:0,y:0}),i=O(et(t),$t(t)).pipe(K());return z([i,n]).pipe(m(([a,s])=>{let{x:p,y:c}=Ve(t),l=ce(t),f=t.closest("table");return f&&t.parentElement&&(p+=f.offsetLeft+t.parentElement.offsetLeft,c+=f.offsetTop+t.parentElement.offsetTop),{active:a,offset:{x:p-s.x+l.width/2-r/2,y:c-s.y+l.height+8}}}))}function Qn(e){let t=e.title;if(!t.length)return S;let r=`__tooltip_${Ga++}`,o=Rt(r,"inline"),n=R(".md-typeset",o);return n.innerHTML=t,C(()=>{let i=new g;return i.subscribe({next({offset:a}){o.style.setProperty("--md-tooltip-x",`${a.x}px`),o.style.setProperty("--md-tooltip-y",`${a.y}px`)},complete(){o.style.removeProperty("--md-tooltip-x"),o.style.removeProperty("--md-tooltip-y")}}),O(i.pipe(b(({active:a})=>a)),i.pipe(_e(250),b(({active:a})=>!a))).subscribe({next({active:a}){a?(e.insertAdjacentElement("afterend",o),e.setAttribute("aria-describedby",r),e.removeAttribute("title")):(o.remove(),e.removeAttribute("aria-describedby"),e.setAttribute("title",t))},complete(){o.remove(),e.removeAttribute("aria-describedby"),e.setAttribute("title",t)}}),i.pipe(Me(16,me)).subscribe(({active:a})=>{o.classList.toggle("md-tooltip--active",a)}),i.pipe(pt(125,me),b(()=>!!e.offsetParent),m(()=>e.offsetParent.getBoundingClientRect()),m(({x:a})=>a)).subscribe({next(a){a?o.style.setProperty("--md-tooltip-0",`${-a}px`):o.style.removeProperty("--md-tooltip-0")},complete(){o.style.removeProperty("--md-tooltip-0")}}),Ja(o,e).pipe(w(a=>i.next(a)),_(()=>i.complete()),m(a=>$({ref:e},a)))}).pipe(Ke(se))}function Xa({viewport$:e}){if(!B("header.autohide"))return I(!1);let t=e.pipe(m(({offset:{y:n}})=>n),Be(2,1),m(([n,i])=>[nMath.abs(i-n.y)>100),m(([,[n]])=>n),K()),o=ze("search");return z([e,o]).pipe(m(([{offset:n},i])=>n.y>400&&!i),K(),v(n=>n?r:I(!1)),Q(!1))}function Kn(e,t){return C(()=>z([ge(e),Xa(t)])).pipe(m(([{height:r},o])=>({height:r,hidden:o})),K((r,o)=>r.height===o.height&&r.hidden===o.hidden),G(1))}function Yn(e,{header$:t,main$:r}){return C(()=>{let o=new g,n=o.pipe(Z(),ie(!0));o.pipe(ee("active"),He(t)).subscribe(([{active:a},{hidden:s}])=>{e.classList.toggle("md-header--shadow",a&&!s),e.hidden=s});let i=ue(P("[title]",e)).pipe(b(()=>B("content.tooltips")),ne(a=>Qn(a)));return r.subscribe(o),t.pipe(W(n),m(a=>$({ref:e},a)),Re(i.pipe(W(n))))})}function Za(e,{viewport$:t,header$:r}){return mr(e,{viewport$:t,header$:r}).pipe(m(({offset:{y:o}})=>{let{height:n}=ce(e);return{active:o>=n}}),ee("active"))}function Bn(e,t){return C(()=>{let r=new g;r.subscribe({next({active:n}){e.classList.toggle("md-header__title--active",n)},complete(){e.classList.remove("md-header__title--active")}});let o=fe(".md-content h1");return typeof o=="undefined"?S:Za(o,t).pipe(w(n=>r.next(n)),_(()=>r.complete()),m(n=>$({ref:e},n)))})}function Gn(e,{viewport$:t,header$:r}){let o=r.pipe(m(({height:i})=>i),K()),n=o.pipe(v(()=>ge(e).pipe(m(({height:i})=>({top:e.offsetTop,bottom:e.offsetTop+i})),ee("bottom"))));return z([o,n,t]).pipe(m(([i,{top:a,bottom:s},{offset:{y:p},size:{height:c}}])=>(c=Math.max(0,c-Math.max(0,a-p,i)-Math.max(0,c+p-s)),{offset:a-i,height:c,active:a-i<=p})),K((i,a)=>i.offset===a.offset&&i.height===a.height&&i.active===a.active))}function es(e){let t=__md_get("__palette")||{index:e.findIndex(o=>matchMedia(o.getAttribute("data-md-color-media")).matches)},r=Math.max(0,Math.min(t.index,e.length-1));return I(...e).pipe(ne(o=>h(o,"change").pipe(m(()=>o))),Q(e[r]),m(o=>({index:e.indexOf(o),color:{media:o.getAttribute("data-md-color-media"),scheme:o.getAttribute("data-md-color-scheme"),primary:o.getAttribute("data-md-color-primary"),accent:o.getAttribute("data-md-color-accent")}})),G(1))}function Jn(e){let t=P("input",e),r=x("meta",{name:"theme-color"});document.head.appendChild(r);let o=x("meta",{name:"color-scheme"});document.head.appendChild(o);let n=Pt("(prefers-color-scheme: light)");return C(()=>{let i=new g;return i.subscribe(a=>{if(document.body.setAttribute("data-md-color-switching",""),a.color.media==="(prefers-color-scheme)"){let s=matchMedia("(prefers-color-scheme: light)"),p=document.querySelector(s.matches?"[data-md-color-media='(prefers-color-scheme: light)']":"[data-md-color-media='(prefers-color-scheme: dark)']");a.color.scheme=p.getAttribute("data-md-color-scheme"),a.color.primary=p.getAttribute("data-md-color-primary"),a.color.accent=p.getAttribute("data-md-color-accent")}for(let[s,p]of Object.entries(a.color))document.body.setAttribute(`data-md-color-${s}`,p);for(let s=0;sa.key==="Enter"),re(i,(a,s)=>s)).subscribe(({index:a})=>{a=(a+1)%t.length,t[a].click(),t[a].focus()}),i.pipe(m(()=>{let a=Se("header"),s=window.getComputedStyle(a);return o.content=s.colorScheme,s.backgroundColor.match(/\d+/g).map(p=>(+p).toString(16).padStart(2,"0")).join("")})).subscribe(a=>r.content=`#${a}`),i.pipe(ve(se)).subscribe(()=>{document.body.removeAttribute("data-md-color-switching")}),es(t).pipe(W(n.pipe(Ce(1))),ct(),w(a=>i.next(a)),_(()=>i.complete()),m(a=>$({ref:e},a)))})}function Xn(e,{progress$:t}){return C(()=>{let r=new g;return r.subscribe(({value:o})=>{e.style.setProperty("--md-progress-value",`${o}`)}),t.pipe(w(o=>r.next({value:o})),_(()=>r.complete()),m(o=>({ref:e,value:o})))})}var Jr=Mt(Br());function ts(e){e.setAttribute("data-md-copying","");let t=e.closest("[data-copy]"),r=t?t.getAttribute("data-copy"):e.innerText;return e.removeAttribute("data-md-copying"),r.trimEnd()}function Zn({alert$:e}){Jr.default.isSupported()&&new j(t=>{new Jr.default("[data-clipboard-target], [data-clipboard-text]",{text:r=>r.getAttribute("data-clipboard-text")||ts(R(r.getAttribute("data-clipboard-target")))}).on("success",r=>t.next(r))}).pipe(w(t=>{t.trigger.focus()}),m(()=>Ee("clipboard.copied"))).subscribe(e)}function ei(e,t){return e.protocol=t.protocol,e.hostname=t.hostname,e}function rs(e,t){let r=new Map;for(let o of P("url",e)){let n=R("loc",o),i=[ei(new URL(n.textContent),t)];r.set(`${i[0]}`,i);for(let a of P("[rel=alternate]",o)){let s=a.getAttribute("href");s!=null&&i.push(ei(new URL(s),t))}}return r}function ur(e){return un(new URL("sitemap.xml",e)).pipe(m(t=>rs(t,new URL(e))),de(()=>I(new Map)))}function os(e,t){if(!(e.target instanceof Element))return S;let r=e.target.closest("a");if(r===null)return S;if(r.target||e.metaKey||e.ctrlKey)return S;let o=new URL(r.href);return o.search=o.hash="",t.has(`${o}`)?(e.preventDefault(),I(new URL(r.href))):S}function ti(e){let t=new Map;for(let r of P(":scope > *",e.head))t.set(r.outerHTML,r);return t}function ri(e){for(let t of P("[href], [src]",e))for(let r of["href","src"]){let o=t.getAttribute(r);if(o&&!/^(?:[a-z]+:)?\/\//i.test(o)){t[r]=t[r];break}}return I(e)}function ns(e){for(let o of["[data-md-component=announce]","[data-md-component=container]","[data-md-component=header-topic]","[data-md-component=outdated]","[data-md-component=logo]","[data-md-component=skip]",...B("navigation.tabs.sticky")?["[data-md-component=tabs]"]:[]]){let n=fe(o),i=fe(o,e);typeof n!="undefined"&&typeof i!="undefined"&&n.replaceWith(i)}let t=ti(document);for(let[o,n]of ti(e))t.has(o)?t.delete(o):document.head.appendChild(n);for(let o of t.values()){let n=o.getAttribute("name");n!=="theme-color"&&n!=="color-scheme"&&o.remove()}let r=Se("container");return We(P("script",r)).pipe(v(o=>{let n=e.createElement("script");if(o.src){for(let i of o.getAttributeNames())n.setAttribute(i,o.getAttribute(i));return o.replaceWith(n),new j(i=>{n.onload=()=>i.complete()})}else return n.textContent=o.textContent,o.replaceWith(n),S}),Z(),ie(document))}function oi({location$:e,viewport$:t,progress$:r}){let o=xe();if(location.protocol==="file:")return S;let n=ur(o.base);I(document).subscribe(ri);let i=h(document.body,"click").pipe(He(n),v(([p,c])=>os(p,c)),pe()),a=h(window,"popstate").pipe(m(ye),pe());i.pipe(re(t)).subscribe(([p,{offset:c}])=>{history.replaceState(c,""),history.pushState(null,"",p)}),O(i,a).subscribe(e);let s=e.pipe(ee("pathname"),v(p=>fn(p,{progress$:r}).pipe(de(()=>(lt(p,!0),S)))),v(ri),v(ns),pe());return O(s.pipe(re(e,(p,c)=>c)),s.pipe(v(()=>e),ee("pathname"),v(()=>e),ee("hash")),e.pipe(K((p,c)=>p.pathname===c.pathname&&p.hash===c.hash),v(()=>i),w(()=>history.back()))).subscribe(p=>{var c,l;history.state!==null||!p.hash?window.scrollTo(0,(l=(c=history.state)==null?void 0:c.y)!=null?l:0):(history.scrollRestoration="auto",pn(p.hash),history.scrollRestoration="manual")}),e.subscribe(()=>{history.scrollRestoration="manual"}),h(window,"beforeunload").subscribe(()=>{history.scrollRestoration="auto"}),t.pipe(ee("offset"),_e(100)).subscribe(({offset:p})=>{history.replaceState(p,"")}),s}var ni=Mt(qr());function ii(e){let t=e.separator.split("|").map(n=>n.replace(/(\(\?[!=<][^)]+\))/g,"").length===0?"\uFFFD":n).join("|"),r=new RegExp(t,"img"),o=(n,i,a)=>`${i}${a}`;return n=>{n=n.replace(/[\s*+\-:~^]+/g," ").trim();let i=new RegExp(`(^|${e.separator}|)(${n.replace(/[|\\{}()[\]^$+*?.-]/g,"\\$&").replace(r,"|")})`,"img");return a=>(0,ni.default)(a).replace(i,o).replace(/<\/mark>(\s+)]*>/img,"$1")}}function jt(e){return e.type===1}function dr(e){return e.type===3}function ai(e,t){let r=yn(e);return O(I(location.protocol!=="file:"),ze("search")).pipe(Ae(o=>o),v(()=>t)).subscribe(({config:o,docs:n})=>r.next({type:0,data:{config:o,docs:n,options:{suggest:B("search.suggest")}}})),r}function si(e){var l;let{selectedVersionSitemap:t,selectedVersionBaseURL:r,currentLocation:o,currentBaseURL:n}=e,i=(l=Xr(n))==null?void 0:l.pathname;if(i===void 0)return;let a=ss(o.pathname,i);if(a===void 0)return;let s=ps(t.keys());if(!t.has(s))return;let p=Xr(a,s);if(!p||!t.has(p.href))return;let c=Xr(a,r);if(c)return c.hash=o.hash,c.search=o.search,c}function Xr(e,t){try{return new URL(e,t)}catch(r){return}}function ss(e,t){if(e.startsWith(t))return e.slice(t.length)}function cs(e,t){let r=Math.min(e.length,t.length),o;for(o=0;oS)),o=r.pipe(m(n=>{let[,i]=t.base.match(/([^/]+)\/?$/);return n.find(({version:a,aliases:s})=>a===i||s.includes(i))||n[0]}));r.pipe(m(n=>new Map(n.map(i=>[`${new URL(`../${i.version}/`,t.base)}`,i]))),v(n=>h(document.body,"click").pipe(b(i=>!i.metaKey&&!i.ctrlKey),re(o),v(([i,a])=>{if(i.target instanceof Element){let s=i.target.closest("a");if(s&&!s.target&&n.has(s.href)){let p=s.href;return!i.target.closest(".md-version")&&n.get(p)===a?S:(i.preventDefault(),I(new URL(p)))}}return S}),v(i=>ur(i).pipe(m(a=>{var s;return(s=si({selectedVersionSitemap:a,selectedVersionBaseURL:i,currentLocation:ye(),currentBaseURL:t.base}))!=null?s:i})))))).subscribe(n=>lt(n,!0)),z([r,o]).subscribe(([n,i])=>{R(".md-header__topic").appendChild(Cn(n,i))}),e.pipe(v(()=>o)).subscribe(n=>{var a;let i=__md_get("__outdated",sessionStorage);if(i===null){i=!0;let s=((a=t.version)==null?void 0:a.default)||"latest";Array.isArray(s)||(s=[s]);e:for(let p of s)for(let c of n.aliases.concat(n.version))if(new RegExp(p,"i").test(c)){i=!1;break e}__md_set("__outdated",i,sessionStorage)}if(i)for(let s of ae("outdated"))s.hidden=!1})}function ls(e,{worker$:t}){let{searchParams:r}=ye();r.has("q")&&(Je("search",!0),e.value=r.get("q"),e.focus(),ze("search").pipe(Ae(i=>!i)).subscribe(()=>{let i=ye();i.searchParams.delete("q"),history.replaceState({},"",`${i}`)}));let o=et(e),n=O(t.pipe(Ae(jt)),h(e,"keyup"),o).pipe(m(()=>e.value),K());return z([n,o]).pipe(m(([i,a])=>({value:i,focus:a})),G(1))}function pi(e,{worker$:t}){let r=new g,o=r.pipe(Z(),ie(!0));z([t.pipe(Ae(jt)),r],(i,a)=>a).pipe(ee("value")).subscribe(({value:i})=>t.next({type:2,data:i})),r.pipe(ee("focus")).subscribe(({focus:i})=>{i&&Je("search",i)}),h(e.form,"reset").pipe(W(o)).subscribe(()=>e.focus());let n=R("header [for=__search]");return h(n,"click").subscribe(()=>e.focus()),ls(e,{worker$:t}).pipe(w(i=>r.next(i)),_(()=>r.complete()),m(i=>$({ref:e},i)),G(1))}function li(e,{worker$:t,query$:r}){let o=new g,n=on(e.parentElement).pipe(b(Boolean)),i=e.parentElement,a=R(":scope > :first-child",e),s=R(":scope > :last-child",e);ze("search").subscribe(l=>s.setAttribute("role",l?"list":"presentation")),o.pipe(re(r),Wr(t.pipe(Ae(jt)))).subscribe(([{items:l},{value:f}])=>{switch(l.length){case 0:a.textContent=f.length?Ee("search.result.none"):Ee("search.result.placeholder");break;case 1:a.textContent=Ee("search.result.one");break;default:let u=sr(l.length);a.textContent=Ee("search.result.other",u)}});let p=o.pipe(w(()=>s.innerHTML=""),v(({items:l})=>O(I(...l.slice(0,10)),I(...l.slice(10)).pipe(Be(4),Vr(n),v(([f])=>f)))),m(Mn),pe());return p.subscribe(l=>s.appendChild(l)),p.pipe(ne(l=>{let f=fe("details",l);return typeof f=="undefined"?S:h(f,"toggle").pipe(W(o),m(()=>f))})).subscribe(l=>{l.open===!1&&l.offsetTop<=i.scrollTop&&i.scrollTo({top:l.offsetTop})}),t.pipe(b(dr),m(({data:l})=>l)).pipe(w(l=>o.next(l)),_(()=>o.complete()),m(l=>$({ref:e},l)))}function ms(e,{query$:t}){return t.pipe(m(({value:r})=>{let o=ye();return o.hash="",r=r.replace(/\s+/g,"+").replace(/&/g,"%26").replace(/=/g,"%3D"),o.search=`q=${r}`,{url:o}}))}function mi(e,t){let r=new g,o=r.pipe(Z(),ie(!0));return r.subscribe(({url:n})=>{e.setAttribute("data-clipboard-text",e.href),e.href=`${n}`}),h(e,"click").pipe(W(o)).subscribe(n=>n.preventDefault()),ms(e,t).pipe(w(n=>r.next(n)),_(()=>r.complete()),m(n=>$({ref:e},n)))}function fi(e,{worker$:t,keyboard$:r}){let o=new g,n=Se("search-query"),i=O(h(n,"keydown"),h(n,"focus")).pipe(ve(se),m(()=>n.value),K());return o.pipe(He(i),m(([{suggest:s},p])=>{let c=p.split(/([\s-]+)/);if(s!=null&&s.length&&c[c.length-1]){let l=s[s.length-1];l.startsWith(c[c.length-1])&&(c[c.length-1]=l)}else c.length=0;return c})).subscribe(s=>e.innerHTML=s.join("").replace(/\s/g," ")),r.pipe(b(({mode:s})=>s==="search")).subscribe(s=>{switch(s.type){case"ArrowRight":e.innerText.length&&n.selectionStart===n.value.length&&(n.value=e.innerText);break}}),t.pipe(b(dr),m(({data:s})=>s)).pipe(w(s=>o.next(s)),_(()=>o.complete()),m(()=>({ref:e})))}function ui(e,{index$:t,keyboard$:r}){let o=xe();try{let n=ai(o.search,t),i=Se("search-query",e),a=Se("search-result",e);h(e,"click").pipe(b(({target:p})=>p instanceof Element&&!!p.closest("a"))).subscribe(()=>Je("search",!1)),r.pipe(b(({mode:p})=>p==="search")).subscribe(p=>{let c=Ie();switch(p.type){case"Enter":if(c===i){let l=new Map;for(let f of P(":first-child [href]",a)){let u=f.firstElementChild;l.set(f,parseFloat(u.getAttribute("data-md-score")))}if(l.size){let[[f]]=[...l].sort(([,u],[,d])=>d-u);f.click()}p.claim()}break;case"Escape":case"Tab":Je("search",!1),i.blur();break;case"ArrowUp":case"ArrowDown":if(typeof c=="undefined")i.focus();else{let l=[i,...P(":not(details) > [href], summary, details[open] [href]",a)],f=Math.max(0,(Math.max(0,l.indexOf(c))+l.length+(p.type==="ArrowUp"?-1:1))%l.length);l[f].focus()}p.claim();break;default:i!==Ie()&&i.focus()}}),r.pipe(b(({mode:p})=>p==="global")).subscribe(p=>{switch(p.type){case"f":case"s":case"/":i.focus(),i.select(),p.claim();break}});let s=pi(i,{worker$:n});return O(s,li(a,{worker$:n,query$:s})).pipe(Re(...ae("search-share",e).map(p=>mi(p,{query$:s})),...ae("search-suggest",e).map(p=>fi(p,{worker$:n,keyboard$:r}))))}catch(n){return e.hidden=!0,Ye}}function di(e,{index$:t,location$:r}){return z([t,r.pipe(Q(ye()),b(o=>!!o.searchParams.get("h")))]).pipe(m(([o,n])=>ii(o.config)(n.searchParams.get("h"))),m(o=>{var a;let n=new Map,i=document.createNodeIterator(e,NodeFilter.SHOW_TEXT);for(let s=i.nextNode();s;s=i.nextNode())if((a=s.parentElement)!=null&&a.offsetHeight){let p=s.textContent,c=o(p);c.length>p.length&&n.set(s,c)}for(let[s,p]of n){let{childNodes:c}=x("span",null,p);s.replaceWith(...Array.from(c))}return{ref:e,nodes:n}}))}function fs(e,{viewport$:t,main$:r}){let o=e.closest(".md-grid"),n=o.offsetTop-o.parentElement.offsetTop;return z([r,t]).pipe(m(([{offset:i,height:a},{offset:{y:s}}])=>(a=a+Math.min(n,Math.max(0,s-i))-n,{height:a,locked:s>=i+n})),K((i,a)=>i.height===a.height&&i.locked===a.locked))}function Zr(e,o){var n=o,{header$:t}=n,r=so(n,["header$"]);let i=R(".md-sidebar__scrollwrap",e),{y:a}=Ve(i);return C(()=>{let s=new g,p=s.pipe(Z(),ie(!0)),c=s.pipe(Me(0,me));return c.pipe(re(t)).subscribe({next([{height:l},{height:f}]){i.style.height=`${l-2*a}px`,e.style.top=`${f}px`},complete(){i.style.height="",e.style.top=""}}),c.pipe(Ae()).subscribe(()=>{for(let l of P(".md-nav__link--active[href]",e)){if(!l.clientHeight)continue;let f=l.closest(".md-sidebar__scrollwrap");if(typeof f!="undefined"){let u=l.offsetTop-f.offsetTop,{height:d}=ce(f);f.scrollTo({top:u-d/2})}}}),ue(P("label[tabindex]",e)).pipe(ne(l=>h(l,"click").pipe(ve(se),m(()=>l),W(p)))).subscribe(l=>{let f=R(`[id="${l.htmlFor}"]`);R(`[aria-labelledby="${l.id}"]`).setAttribute("aria-expanded",`${f.checked}`)}),fs(e,r).pipe(w(l=>s.next(l)),_(()=>s.complete()),m(l=>$({ref:e},l)))})}function hi(e,t){if(typeof t!="undefined"){let r=`https://api.github.com/repos/${e}/${t}`;return st(je(`${r}/releases/latest`).pipe(de(()=>S),m(o=>({version:o.tag_name})),De({})),je(r).pipe(de(()=>S),m(o=>({stars:o.stargazers_count,forks:o.forks_count})),De({}))).pipe(m(([o,n])=>$($({},o),n)))}else{let r=`https://api.github.com/users/${e}`;return je(r).pipe(m(o=>({repositories:o.public_repos})),De({}))}}function bi(e,t){let r=`https://${e}/api/v4/projects/${encodeURIComponent(t)}`;return st(je(`${r}/releases/permalink/latest`).pipe(de(()=>S),m(({tag_name:o})=>({version:o})),De({})),je(r).pipe(de(()=>S),m(({star_count:o,forks_count:n})=>({stars:o,forks:n})),De({}))).pipe(m(([o,n])=>$($({},o),n)))}function vi(e){let t=e.match(/^.+github\.com\/([^/]+)\/?([^/]+)?/i);if(t){let[,r,o]=t;return hi(r,o)}if(t=e.match(/^.+?([^/]*gitlab[^/]+)\/(.+?)\/?$/i),t){let[,r,o]=t;return bi(r,o)}return S}var us;function ds(e){return us||(us=C(()=>{let t=__md_get("__source",sessionStorage);if(t)return I(t);if(ae("consent").length){let o=__md_get("__consent");if(!(o&&o.github))return S}return vi(e.href).pipe(w(o=>__md_set("__source",o,sessionStorage)))}).pipe(de(()=>S),b(t=>Object.keys(t).length>0),m(t=>({facts:t})),G(1)))}function gi(e){let t=R(":scope > :last-child",e);return C(()=>{let r=new g;return r.subscribe(({facts:o})=>{t.appendChild(_n(o)),t.classList.add("md-source__repository--active")}),ds(e).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}function hs(e,{viewport$:t,header$:r}){return ge(document.body).pipe(v(()=>mr(e,{header$:r,viewport$:t})),m(({offset:{y:o}})=>({hidden:o>=10})),ee("hidden"))}function yi(e,t){return C(()=>{let r=new g;return r.subscribe({next({hidden:o}){e.hidden=o},complete(){e.hidden=!1}}),(B("navigation.tabs.sticky")?I({hidden:!1}):hs(e,t)).pipe(w(o=>r.next(o)),_(()=>r.complete()),m(o=>$({ref:e},o)))})}function bs(e,{viewport$:t,header$:r}){let o=new Map,n=P(".md-nav__link",e);for(let s of n){let p=decodeURIComponent(s.hash.substring(1)),c=fe(`[id="${p}"]`);typeof c!="undefined"&&o.set(s,c)}let i=r.pipe(ee("height"),m(({height:s})=>{let p=Se("main"),c=R(":scope > :first-child",p);return s+.8*(c.offsetTop-p.offsetTop)}),pe());return ge(document.body).pipe(ee("height"),v(s=>C(()=>{let p=[];return I([...o].reduce((c,[l,f])=>{for(;p.length&&o.get(p[p.length-1]).tagName>=f.tagName;)p.pop();let u=f.offsetTop;for(;!u&&f.parentElement;)f=f.parentElement,u=f.offsetTop;let d=f.offsetParent;for(;d;d=d.offsetParent)u+=d.offsetTop;return c.set([...p=[...p,l]].reverse(),u)},new Map))}).pipe(m(p=>new Map([...p].sort(([,c],[,l])=>c-l))),He(i),v(([p,c])=>t.pipe(Fr(([l,f],{offset:{y:u},size:d})=>{let y=u+d.height>=Math.floor(s.height);for(;f.length;){let[,L]=f[0];if(L-c=u&&!y)f=[l.pop(),...f];else break}return[l,f]},[[],[...p]]),K((l,f)=>l[0]===f[0]&&l[1]===f[1])))))).pipe(m(([s,p])=>({prev:s.map(([c])=>c),next:p.map(([c])=>c)})),Q({prev:[],next:[]}),Be(2,1),m(([s,p])=>s.prev.length{let i=new g,a=i.pipe(Z(),ie(!0));if(i.subscribe(({prev:s,next:p})=>{for(let[c]of p)c.classList.remove("md-nav__link--passed"),c.classList.remove("md-nav__link--active");for(let[c,[l]]of s.entries())l.classList.add("md-nav__link--passed"),l.classList.toggle("md-nav__link--active",c===s.length-1)}),B("toc.follow")){let s=O(t.pipe(_e(1),m(()=>{})),t.pipe(_e(250),m(()=>"smooth")));i.pipe(b(({prev:p})=>p.length>0),He(o.pipe(ve(se))),re(s)).subscribe(([[{prev:p}],c])=>{let[l]=p[p.length-1];if(l.offsetHeight){let f=cr(l);if(typeof f!="undefined"){let u=l.offsetTop-f.offsetTop,{height:d}=ce(f);f.scrollTo({top:u-d/2,behavior:c})}}})}return B("navigation.tracking")&&t.pipe(W(a),ee("offset"),_e(250),Ce(1),W(n.pipe(Ce(1))),ct({delay:250}),re(i)).subscribe(([,{prev:s}])=>{let p=ye(),c=s[s.length-1];if(c&&c.length){let[l]=c,{hash:f}=new URL(l.href);p.hash!==f&&(p.hash=f,history.replaceState({},"",`${p}`))}else p.hash="",history.replaceState({},"",`${p}`)}),bs(e,{viewport$:t,header$:r}).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))})}function vs(e,{viewport$:t,main$:r,target$:o}){let n=t.pipe(m(({offset:{y:a}})=>a),Be(2,1),m(([a,s])=>a>s&&s>0),K()),i=r.pipe(m(({active:a})=>a));return z([i,n]).pipe(m(([a,s])=>!(a&&s)),K(),W(o.pipe(Ce(1))),ie(!0),ct({delay:250}),m(a=>({hidden:a})))}function Ei(e,{viewport$:t,header$:r,main$:o,target$:n}){let i=new g,a=i.pipe(Z(),ie(!0));return i.subscribe({next({hidden:s}){e.hidden=s,s?(e.setAttribute("tabindex","-1"),e.blur()):e.removeAttribute("tabindex")},complete(){e.style.top="",e.hidden=!0,e.removeAttribute("tabindex")}}),r.pipe(W(a),ee("height")).subscribe(({height:s})=>{e.style.top=`${s+16}px`}),h(e,"click").subscribe(s=>{s.preventDefault(),window.scrollTo({top:0})}),vs(e,{viewport$:t,main$:o,target$:n}).pipe(w(s=>i.next(s)),_(()=>i.complete()),m(s=>$({ref:e},s)))}function wi({document$:e,viewport$:t}){e.pipe(v(()=>P(".md-ellipsis")),ne(r=>tt(r).pipe(W(e.pipe(Ce(1))),b(o=>o),m(()=>r),Te(1))),b(r=>r.offsetWidth{let o=r.innerText,n=r.closest("a")||r;return n.title=o,B("content.tooltips")?mt(n,{viewport$:t}).pipe(W(e.pipe(Ce(1))),_(()=>n.removeAttribute("title"))):S})).subscribe(),B("content.tooltips")&&e.pipe(v(()=>P(".md-status")),ne(r=>mt(r,{viewport$:t}))).subscribe()}function Ti({document$:e,tablet$:t}){e.pipe(v(()=>P(".md-toggle--indeterminate")),w(r=>{r.indeterminate=!0,r.checked=!1}),ne(r=>h(r,"change").pipe(Dr(()=>r.classList.contains("md-toggle--indeterminate")),m(()=>r))),re(t)).subscribe(([r,o])=>{r.classList.remove("md-toggle--indeterminate"),o&&(r.checked=!1)})}function gs(){return/(iPad|iPhone|iPod)/.test(navigator.userAgent)}function Si({document$:e}){e.pipe(v(()=>P("[data-md-scrollfix]")),w(t=>t.removeAttribute("data-md-scrollfix")),b(gs),ne(t=>h(t,"touchstart").pipe(m(()=>t)))).subscribe(t=>{let r=t.scrollTop;r===0?t.scrollTop=1:r+t.offsetHeight===t.scrollHeight&&(t.scrollTop=r-1)})}function Oi({viewport$:e,tablet$:t}){z([ze("search"),t]).pipe(m(([r,o])=>r&&!o),v(r=>I(r).pipe(Ge(r?400:100))),re(e)).subscribe(([r,{offset:{y:o}}])=>{if(r)document.body.setAttribute("data-md-scrolllock",""),document.body.style.top=`-${o}px`;else{let n=-1*parseInt(document.body.style.top,10);document.body.removeAttribute("data-md-scrolllock"),document.body.style.top="",n&&window.scrollTo(0,n)}})}Object.entries||(Object.entries=function(e){let t=[];for(let r of Object.keys(e))t.push([r,e[r]]);return t});Object.values||(Object.values=function(e){let t=[];for(let r of Object.keys(e))t.push(e[r]);return t});typeof Element!="undefined"&&(Element.prototype.scrollTo||(Element.prototype.scrollTo=function(e,t){typeof e=="object"?(this.scrollLeft=e.left,this.scrollTop=e.top):(this.scrollLeft=e,this.scrollTop=t)}),Element.prototype.replaceWith||(Element.prototype.replaceWith=function(...e){let t=this.parentNode;if(t){e.length===0&&t.removeChild(this);for(let r=e.length-1;r>=0;r--){let o=e[r];typeof o=="string"?o=document.createTextNode(o):o.parentNode&&o.parentNode.removeChild(o),r?t.insertBefore(this.previousSibling,o):t.replaceChild(o,this)}}}));function ys(){return location.protocol==="file:"?Tt(`${new URL("search/search_index.js",eo.base)}`).pipe(m(()=>__index),G(1)):je(new URL("search/search_index.json",eo.base))}document.documentElement.classList.remove("no-js");document.documentElement.classList.add("js");var ot=Go(),Ut=sn(),Lt=ln(Ut),to=an(),Oe=gn(),hr=Pt("(min-width: 960px)"),Mi=Pt("(min-width: 1220px)"),_i=mn(),eo=xe(),Ai=document.forms.namedItem("search")?ys():Ye,ro=new g;Zn({alert$:ro});var oo=new g;B("navigation.instant")&&oi({location$:Ut,viewport$:Oe,progress$:oo}).subscribe(ot);var Li;((Li=eo.version)==null?void 0:Li.provider)==="mike"&&ci({document$:ot});O(Ut,Lt).pipe(Ge(125)).subscribe(()=>{Je("drawer",!1),Je("search",!1)});to.pipe(b(({mode:e})=>e==="global")).subscribe(e=>{switch(e.type){case"p":case",":let t=fe("link[rel=prev]");typeof t!="undefined"&<(t);break;case"n":case".":let r=fe("link[rel=next]");typeof r!="undefined"&<(r);break;case"Enter":let o=Ie();o instanceof HTMLLabelElement&&o.click()}});wi({viewport$:Oe,document$:ot});Ti({document$:ot,tablet$:hr});Si({document$:ot});Oi({viewport$:Oe,tablet$:hr});var rt=Kn(Se("header"),{viewport$:Oe}),Ft=ot.pipe(m(()=>Se("main")),v(e=>Gn(e,{viewport$:Oe,header$:rt})),G(1)),xs=O(...ae("consent").map(e=>En(e,{target$:Lt})),...ae("dialog").map(e=>qn(e,{alert$:ro})),...ae("palette").map(e=>Jn(e)),...ae("progress").map(e=>Xn(e,{progress$:oo})),...ae("search").map(e=>ui(e,{index$:Ai,keyboard$:to})),...ae("source").map(e=>gi(e))),Es=C(()=>O(...ae("announce").map(e=>xn(e)),...ae("content").map(e=>zn(e,{viewport$:Oe,target$:Lt,print$:_i})),...ae("content").map(e=>B("search.highlight")?di(e,{index$:Ai,location$:Ut}):S),...ae("header").map(e=>Yn(e,{viewport$:Oe,header$:rt,main$:Ft})),...ae("header-title").map(e=>Bn(e,{viewport$:Oe,header$:rt})),...ae("sidebar").map(e=>e.getAttribute("data-md-type")==="navigation"?Nr(Mi,()=>Zr(e,{viewport$:Oe,header$:rt,main$:Ft})):Nr(hr,()=>Zr(e,{viewport$:Oe,header$:rt,main$:Ft}))),...ae("tabs").map(e=>yi(e,{viewport$:Oe,header$:rt})),...ae("toc").map(e=>xi(e,{viewport$:Oe,header$:rt,main$:Ft,target$:Lt})),...ae("top").map(e=>Ei(e,{viewport$:Oe,header$:rt,main$:Ft,target$:Lt})))),Ci=ot.pipe(v(()=>Es),Re(xs),G(1));Ci.subscribe();window.document$=ot;window.location$=Ut;window.target$=Lt;window.keyboard$=to;window.viewport$=Oe;window.tablet$=hr;window.screen$=Mi;window.print$=_i;window.alert$=ro;window.progress$=oo;window.component$=Ci;})(); +//# sourceMappingURL=bundle.88dd0f4e.min.js.map + diff --git a/assets/javascripts/bundle.88dd0f4e.min.js.map b/assets/javascripts/bundle.88dd0f4e.min.js.map new file mode 100644 index 0000000..dab2a87 --- /dev/null +++ b/assets/javascripts/bundle.88dd0f4e.min.js.map @@ -0,0 +1,7 @@ +{ + "version": 3, + "sources": ["node_modules/focus-visible/dist/focus-visible.js", "node_modules/escape-html/index.js", "node_modules/clipboard/dist/clipboard.js", "src/templates/assets/javascripts/bundle.ts", "node_modules/tslib/tslib.es6.mjs", "node_modules/rxjs/src/internal/util/isFunction.ts", "node_modules/rxjs/src/internal/util/createErrorClass.ts", "node_modules/rxjs/src/internal/util/UnsubscriptionError.ts", "node_modules/rxjs/src/internal/util/arrRemove.ts", "node_modules/rxjs/src/internal/Subscription.ts", "node_modules/rxjs/src/internal/config.ts", "node_modules/rxjs/src/internal/scheduler/timeoutProvider.ts", "node_modules/rxjs/src/internal/util/reportUnhandledError.ts", "node_modules/rxjs/src/internal/util/noop.ts", "node_modules/rxjs/src/internal/NotificationFactories.ts", "node_modules/rxjs/src/internal/util/errorContext.ts", "node_modules/rxjs/src/internal/Subscriber.ts", "node_modules/rxjs/src/internal/symbol/observable.ts", "node_modules/rxjs/src/internal/util/identity.ts", "node_modules/rxjs/src/internal/util/pipe.ts", "node_modules/rxjs/src/internal/Observable.ts", "node_modules/rxjs/src/internal/util/lift.ts", "node_modules/rxjs/src/internal/operators/OperatorSubscriber.ts", "node_modules/rxjs/src/internal/scheduler/animationFrameProvider.ts", "node_modules/rxjs/src/internal/util/ObjectUnsubscribedError.ts", "node_modules/rxjs/src/internal/Subject.ts", "node_modules/rxjs/src/internal/BehaviorSubject.ts", "node_modules/rxjs/src/internal/scheduler/dateTimestampProvider.ts", "node_modules/rxjs/src/internal/ReplaySubject.ts", "node_modules/rxjs/src/internal/scheduler/Action.ts", "node_modules/rxjs/src/internal/scheduler/intervalProvider.ts", "node_modules/rxjs/src/internal/scheduler/AsyncAction.ts", "node_modules/rxjs/src/internal/Scheduler.ts", "node_modules/rxjs/src/internal/scheduler/AsyncScheduler.ts", "node_modules/rxjs/src/internal/scheduler/async.ts", "node_modules/rxjs/src/internal/scheduler/QueueAction.ts", "node_modules/rxjs/src/internal/scheduler/QueueScheduler.ts", "node_modules/rxjs/src/internal/scheduler/queue.ts", "node_modules/rxjs/src/internal/scheduler/AnimationFrameAction.ts", "node_modules/rxjs/src/internal/scheduler/AnimationFrameScheduler.ts", "node_modules/rxjs/src/internal/scheduler/animationFrame.ts", "node_modules/rxjs/src/internal/observable/empty.ts", "node_modules/rxjs/src/internal/util/isScheduler.ts", "node_modules/rxjs/src/internal/util/args.ts", "node_modules/rxjs/src/internal/util/isArrayLike.ts", "node_modules/rxjs/src/internal/util/isPromise.ts", "node_modules/rxjs/src/internal/util/isInteropObservable.ts", "node_modules/rxjs/src/internal/util/isAsyncIterable.ts", "node_modules/rxjs/src/internal/util/throwUnobservableError.ts", "node_modules/rxjs/src/internal/symbol/iterator.ts", "node_modules/rxjs/src/internal/util/isIterable.ts", "node_modules/rxjs/src/internal/util/isReadableStreamLike.ts", "node_modules/rxjs/src/internal/observable/innerFrom.ts", "node_modules/rxjs/src/internal/util/executeSchedule.ts", "node_modules/rxjs/src/internal/operators/observeOn.ts", "node_modules/rxjs/src/internal/operators/subscribeOn.ts", "node_modules/rxjs/src/internal/scheduled/scheduleObservable.ts", "node_modules/rxjs/src/internal/scheduled/schedulePromise.ts", "node_modules/rxjs/src/internal/scheduled/scheduleArray.ts", "node_modules/rxjs/src/internal/scheduled/scheduleIterable.ts", "node_modules/rxjs/src/internal/scheduled/scheduleAsyncIterable.ts", "node_modules/rxjs/src/internal/scheduled/scheduleReadableStreamLike.ts", "node_modules/rxjs/src/internal/scheduled/scheduled.ts", "node_modules/rxjs/src/internal/observable/from.ts", "node_modules/rxjs/src/internal/observable/of.ts", "node_modules/rxjs/src/internal/observable/throwError.ts", "node_modules/rxjs/src/internal/util/EmptyError.ts", "node_modules/rxjs/src/internal/util/isDate.ts", "node_modules/rxjs/src/internal/operators/map.ts", "node_modules/rxjs/src/internal/util/mapOneOrManyArgs.ts", "node_modules/rxjs/src/internal/util/argsArgArrayOrObject.ts", "node_modules/rxjs/src/internal/util/createObject.ts", "node_modules/rxjs/src/internal/observable/combineLatest.ts", "node_modules/rxjs/src/internal/operators/mergeInternals.ts", "node_modules/rxjs/src/internal/operators/mergeMap.ts", "node_modules/rxjs/src/internal/operators/mergeAll.ts", "node_modules/rxjs/src/internal/operators/concatAll.ts", "node_modules/rxjs/src/internal/observable/concat.ts", "node_modules/rxjs/src/internal/observable/defer.ts", "node_modules/rxjs/src/internal/observable/fromEvent.ts", "node_modules/rxjs/src/internal/observable/fromEventPattern.ts", "node_modules/rxjs/src/internal/observable/timer.ts", "node_modules/rxjs/src/internal/observable/merge.ts", "node_modules/rxjs/src/internal/observable/never.ts", "node_modules/rxjs/src/internal/util/argsOrArgArray.ts", "node_modules/rxjs/src/internal/operators/filter.ts", "node_modules/rxjs/src/internal/observable/zip.ts", "node_modules/rxjs/src/internal/operators/audit.ts", "node_modules/rxjs/src/internal/operators/auditTime.ts", "node_modules/rxjs/src/internal/operators/bufferCount.ts", "node_modules/rxjs/src/internal/operators/catchError.ts", "node_modules/rxjs/src/internal/operators/scanInternals.ts", "node_modules/rxjs/src/internal/operators/combineLatest.ts", "node_modules/rxjs/src/internal/operators/combineLatestWith.ts", "node_modules/rxjs/src/internal/operators/debounce.ts", "node_modules/rxjs/src/internal/operators/debounceTime.ts", "node_modules/rxjs/src/internal/operators/defaultIfEmpty.ts", "node_modules/rxjs/src/internal/operators/take.ts", "node_modules/rxjs/src/internal/operators/ignoreElements.ts", "node_modules/rxjs/src/internal/operators/mapTo.ts", "node_modules/rxjs/src/internal/operators/delayWhen.ts", "node_modules/rxjs/src/internal/operators/delay.ts", "node_modules/rxjs/src/internal/operators/distinctUntilChanged.ts", "node_modules/rxjs/src/internal/operators/distinctUntilKeyChanged.ts", "node_modules/rxjs/src/internal/operators/throwIfEmpty.ts", "node_modules/rxjs/src/internal/operators/endWith.ts", "node_modules/rxjs/src/internal/operators/finalize.ts", "node_modules/rxjs/src/internal/operators/first.ts", "node_modules/rxjs/src/internal/operators/takeLast.ts", "node_modules/rxjs/src/internal/operators/merge.ts", "node_modules/rxjs/src/internal/operators/mergeWith.ts", "node_modules/rxjs/src/internal/operators/repeat.ts", "node_modules/rxjs/src/internal/operators/scan.ts", "node_modules/rxjs/src/internal/operators/share.ts", "node_modules/rxjs/src/internal/operators/shareReplay.ts", "node_modules/rxjs/src/internal/operators/skip.ts", "node_modules/rxjs/src/internal/operators/skipUntil.ts", "node_modules/rxjs/src/internal/operators/startWith.ts", "node_modules/rxjs/src/internal/operators/switchMap.ts", "node_modules/rxjs/src/internal/operators/takeUntil.ts", "node_modules/rxjs/src/internal/operators/takeWhile.ts", "node_modules/rxjs/src/internal/operators/tap.ts", "node_modules/rxjs/src/internal/operators/throttle.ts", "node_modules/rxjs/src/internal/operators/throttleTime.ts", "node_modules/rxjs/src/internal/operators/withLatestFrom.ts", "node_modules/rxjs/src/internal/operators/zip.ts", "node_modules/rxjs/src/internal/operators/zipWith.ts", "src/templates/assets/javascripts/browser/document/index.ts", "src/templates/assets/javascripts/browser/element/_/index.ts", "src/templates/assets/javascripts/browser/element/focus/index.ts", "src/templates/assets/javascripts/browser/element/hover/index.ts", "src/templates/assets/javascripts/utilities/h/index.ts", "src/templates/assets/javascripts/utilities/round/index.ts", "src/templates/assets/javascripts/browser/script/index.ts", "src/templates/assets/javascripts/browser/element/size/_/index.ts", "src/templates/assets/javascripts/browser/element/size/content/index.ts", "src/templates/assets/javascripts/browser/element/offset/_/index.ts", "src/templates/assets/javascripts/browser/element/offset/content/index.ts", "src/templates/assets/javascripts/browser/element/visibility/index.ts", "src/templates/assets/javascripts/browser/toggle/index.ts", "src/templates/assets/javascripts/browser/keyboard/index.ts", "src/templates/assets/javascripts/browser/location/_/index.ts", "src/templates/assets/javascripts/browser/location/hash/index.ts", "src/templates/assets/javascripts/browser/media/index.ts", "src/templates/assets/javascripts/browser/request/index.ts", "src/templates/assets/javascripts/browser/viewport/offset/index.ts", "src/templates/assets/javascripts/browser/viewport/size/index.ts", "src/templates/assets/javascripts/browser/viewport/_/index.ts", "src/templates/assets/javascripts/browser/viewport/at/index.ts", "src/templates/assets/javascripts/browser/worker/index.ts", "src/templates/assets/javascripts/_/index.ts", "src/templates/assets/javascripts/components/_/index.ts", "src/templates/assets/javascripts/components/announce/index.ts", "src/templates/assets/javascripts/components/consent/index.ts", "src/templates/assets/javascripts/templates/tooltip/index.tsx", "src/templates/assets/javascripts/templates/annotation/index.tsx", "src/templates/assets/javascripts/templates/clipboard/index.tsx", "src/templates/assets/javascripts/templates/search/index.tsx", "src/templates/assets/javascripts/templates/source/index.tsx", "src/templates/assets/javascripts/templates/tabbed/index.tsx", "src/templates/assets/javascripts/templates/table/index.tsx", "src/templates/assets/javascripts/templates/version/index.tsx", "src/templates/assets/javascripts/components/tooltip2/index.ts", "src/templates/assets/javascripts/components/content/annotation/_/index.ts", "src/templates/assets/javascripts/components/content/annotation/list/index.ts", "src/templates/assets/javascripts/components/content/annotation/block/index.ts", "src/templates/assets/javascripts/components/content/code/_/index.ts", "src/templates/assets/javascripts/components/content/details/index.ts", "src/templates/assets/javascripts/components/content/mermaid/index.css", "src/templates/assets/javascripts/components/content/mermaid/index.ts", "src/templates/assets/javascripts/components/content/table/index.ts", "src/templates/assets/javascripts/components/content/tabs/index.ts", "src/templates/assets/javascripts/components/content/_/index.ts", "src/templates/assets/javascripts/components/dialog/index.ts", "src/templates/assets/javascripts/components/tooltip/index.ts", "src/templates/assets/javascripts/components/header/_/index.ts", "src/templates/assets/javascripts/components/header/title/index.ts", "src/templates/assets/javascripts/components/main/index.ts", "src/templates/assets/javascripts/components/palette/index.ts", "src/templates/assets/javascripts/components/progress/index.ts", "src/templates/assets/javascripts/integrations/clipboard/index.ts", "src/templates/assets/javascripts/integrations/sitemap/index.ts", "src/templates/assets/javascripts/integrations/instant/index.ts", "src/templates/assets/javascripts/integrations/search/highlighter/index.ts", "src/templates/assets/javascripts/integrations/search/worker/message/index.ts", "src/templates/assets/javascripts/integrations/search/worker/_/index.ts", "src/templates/assets/javascripts/integrations/version/findurl/index.ts", "src/templates/assets/javascripts/integrations/version/index.ts", "src/templates/assets/javascripts/components/search/query/index.ts", "src/templates/assets/javascripts/components/search/result/index.ts", "src/templates/assets/javascripts/components/search/share/index.ts", "src/templates/assets/javascripts/components/search/suggest/index.ts", "src/templates/assets/javascripts/components/search/_/index.ts", "src/templates/assets/javascripts/components/search/highlight/index.ts", "src/templates/assets/javascripts/components/sidebar/index.ts", "src/templates/assets/javascripts/components/source/facts/github/index.ts", "src/templates/assets/javascripts/components/source/facts/gitlab/index.ts", "src/templates/assets/javascripts/components/source/facts/_/index.ts", "src/templates/assets/javascripts/components/source/_/index.ts", "src/templates/assets/javascripts/components/tabs/index.ts", "src/templates/assets/javascripts/components/toc/index.ts", "src/templates/assets/javascripts/components/top/index.ts", "src/templates/assets/javascripts/patches/ellipsis/index.ts", "src/templates/assets/javascripts/patches/indeterminate/index.ts", "src/templates/assets/javascripts/patches/scrollfix/index.ts", "src/templates/assets/javascripts/patches/scrolllock/index.ts", "src/templates/assets/javascripts/polyfills/index.ts"], + "sourcesContent": ["(function (global, factory) {\n typeof exports === 'object' && typeof module !== 'undefined' ? factory() :\n typeof define === 'function' && define.amd ? define(factory) :\n (factory());\n}(this, (function () { 'use strict';\n\n /**\n * Applies the :focus-visible polyfill at the given scope.\n * A scope in this case is either the top-level Document or a Shadow Root.\n *\n * @param {(Document|ShadowRoot)} scope\n * @see https://github.com/WICG/focus-visible\n */\n function applyFocusVisiblePolyfill(scope) {\n var hadKeyboardEvent = true;\n var hadFocusVisibleRecently = false;\n var hadFocusVisibleRecentlyTimeout = null;\n\n var inputTypesAllowlist = {\n text: true,\n search: true,\n url: true,\n tel: true,\n email: true,\n password: true,\n number: true,\n date: true,\n month: true,\n week: true,\n time: true,\n datetime: true,\n 'datetime-local': true\n };\n\n /**\n * Helper function for legacy browsers and iframes which sometimes focus\n * elements like document, body, and non-interactive SVG.\n * @param {Element} el\n */\n function isValidFocusTarget(el) {\n if (\n el &&\n el !== document &&\n el.nodeName !== 'HTML' &&\n el.nodeName !== 'BODY' &&\n 'classList' in el &&\n 'contains' in el.classList\n ) {\n return true;\n }\n return false;\n }\n\n /**\n * Computes whether the given element should automatically trigger the\n * `focus-visible` class being added, i.e. whether it should always match\n * `:focus-visible` when focused.\n * @param {Element} el\n * @return {boolean}\n */\n function focusTriggersKeyboardModality(el) {\n var type = el.type;\n var tagName = el.tagName;\n\n if (tagName === 'INPUT' && inputTypesAllowlist[type] && !el.readOnly) {\n return true;\n }\n\n if (tagName === 'TEXTAREA' && !el.readOnly) {\n return true;\n }\n\n if (el.isContentEditable) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Add the `focus-visible` class to the given element if it was not added by\n * the author.\n * @param {Element} el\n */\n function addFocusVisibleClass(el) {\n if (el.classList.contains('focus-visible')) {\n return;\n }\n el.classList.add('focus-visible');\n el.setAttribute('data-focus-visible-added', '');\n }\n\n /**\n * Remove the `focus-visible` class from the given element if it was not\n * originally added by the author.\n * @param {Element} el\n */\n function removeFocusVisibleClass(el) {\n if (!el.hasAttribute('data-focus-visible-added')) {\n return;\n }\n el.classList.remove('focus-visible');\n el.removeAttribute('data-focus-visible-added');\n }\n\n /**\n * If the most recent user interaction was via the keyboard;\n * and the key press did not include a meta, alt/option, or control key;\n * then the modality is keyboard. Otherwise, the modality is not keyboard.\n * Apply `focus-visible` to any current active element and keep track\n * of our keyboard modality state with `hadKeyboardEvent`.\n * @param {KeyboardEvent} e\n */\n function onKeyDown(e) {\n if (e.metaKey || e.altKey || e.ctrlKey) {\n return;\n }\n\n if (isValidFocusTarget(scope.activeElement)) {\n addFocusVisibleClass(scope.activeElement);\n }\n\n hadKeyboardEvent = true;\n }\n\n /**\n * If at any point a user clicks with a pointing device, ensure that we change\n * the modality away from keyboard.\n * This avoids the situation where a user presses a key on an already focused\n * element, and then clicks on a different element, focusing it with a\n * pointing device, while we still think we're in keyboard modality.\n * @param {Event} e\n */\n function onPointerDown(e) {\n hadKeyboardEvent = false;\n }\n\n /**\n * On `focus`, add the `focus-visible` class to the target if:\n * - the target received focus as a result of keyboard navigation, or\n * - the event target is an element that will likely require interaction\n * via the keyboard (e.g. a text box)\n * @param {Event} e\n */\n function onFocus(e) {\n // Prevent IE from focusing the document or HTML element.\n if (!isValidFocusTarget(e.target)) {\n return;\n }\n\n if (hadKeyboardEvent || focusTriggersKeyboardModality(e.target)) {\n addFocusVisibleClass(e.target);\n }\n }\n\n /**\n * On `blur`, remove the `focus-visible` class from the target.\n * @param {Event} e\n */\n function onBlur(e) {\n if (!isValidFocusTarget(e.target)) {\n return;\n }\n\n if (\n e.target.classList.contains('focus-visible') ||\n e.target.hasAttribute('data-focus-visible-added')\n ) {\n // To detect a tab/window switch, we look for a blur event followed\n // rapidly by a visibility change.\n // If we don't see a visibility change within 100ms, it's probably a\n // regular focus change.\n hadFocusVisibleRecently = true;\n window.clearTimeout(hadFocusVisibleRecentlyTimeout);\n hadFocusVisibleRecentlyTimeout = window.setTimeout(function() {\n hadFocusVisibleRecently = false;\n }, 100);\n removeFocusVisibleClass(e.target);\n }\n }\n\n /**\n * If the user changes tabs, keep track of whether or not the previously\n * focused element had .focus-visible.\n * @param {Event} e\n */\n function onVisibilityChange(e) {\n if (document.visibilityState === 'hidden') {\n // If the tab becomes active again, the browser will handle calling focus\n // on the element (Safari actually calls it twice).\n // If this tab change caused a blur on an element with focus-visible,\n // re-apply the class when the user switches back to the tab.\n if (hadFocusVisibleRecently) {\n hadKeyboardEvent = true;\n }\n addInitialPointerMoveListeners();\n }\n }\n\n /**\n * Add a group of listeners to detect usage of any pointing devices.\n * These listeners will be added when the polyfill first loads, and anytime\n * the window is blurred, so that they are active when the window regains\n * focus.\n */\n function addInitialPointerMoveListeners() {\n document.addEventListener('mousemove', onInitialPointerMove);\n document.addEventListener('mousedown', onInitialPointerMove);\n document.addEventListener('mouseup', onInitialPointerMove);\n document.addEventListener('pointermove', onInitialPointerMove);\n document.addEventListener('pointerdown', onInitialPointerMove);\n document.addEventListener('pointerup', onInitialPointerMove);\n document.addEventListener('touchmove', onInitialPointerMove);\n document.addEventListener('touchstart', onInitialPointerMove);\n document.addEventListener('touchend', onInitialPointerMove);\n }\n\n function removeInitialPointerMoveListeners() {\n document.removeEventListener('mousemove', onInitialPointerMove);\n document.removeEventListener('mousedown', onInitialPointerMove);\n document.removeEventListener('mouseup', onInitialPointerMove);\n document.removeEventListener('pointermove', onInitialPointerMove);\n document.removeEventListener('pointerdown', onInitialPointerMove);\n document.removeEventListener('pointerup', onInitialPointerMove);\n document.removeEventListener('touchmove', onInitialPointerMove);\n document.removeEventListener('touchstart', onInitialPointerMove);\n document.removeEventListener('touchend', onInitialPointerMove);\n }\n\n /**\n * When the polfyill first loads, assume the user is in keyboard modality.\n * If any event is received from a pointing device (e.g. mouse, pointer,\n * touch), turn off keyboard modality.\n * This accounts for situations where focus enters the page from the URL bar.\n * @param {Event} e\n */\n function onInitialPointerMove(e) {\n // Work around a Safari quirk that fires a mousemove on whenever the\n // window blurs, even if you're tabbing out of the page. \u00AF\\_(\u30C4)_/\u00AF\n if (e.target.nodeName && e.target.nodeName.toLowerCase() === 'html') {\n return;\n }\n\n hadKeyboardEvent = false;\n removeInitialPointerMoveListeners();\n }\n\n // For some kinds of state, we are interested in changes at the global scope\n // only. For example, global pointer input, global key presses and global\n // visibility change should affect the state at every scope:\n document.addEventListener('keydown', onKeyDown, true);\n document.addEventListener('mousedown', onPointerDown, true);\n document.addEventListener('pointerdown', onPointerDown, true);\n document.addEventListener('touchstart', onPointerDown, true);\n document.addEventListener('visibilitychange', onVisibilityChange, true);\n\n addInitialPointerMoveListeners();\n\n // For focus and blur, we specifically care about state changes in the local\n // scope. This is because focus / blur events that originate from within a\n // shadow root are not re-dispatched from the host element if it was already\n // the active element in its own scope:\n scope.addEventListener('focus', onFocus, true);\n scope.addEventListener('blur', onBlur, true);\n\n // We detect that a node is a ShadowRoot by ensuring that it is a\n // DocumentFragment and also has a host property. This check covers native\n // implementation and polyfill implementation transparently. If we only cared\n // about the native implementation, we could just check if the scope was\n // an instance of a ShadowRoot.\n if (scope.nodeType === Node.DOCUMENT_FRAGMENT_NODE && scope.host) {\n // Since a ShadowRoot is a special kind of DocumentFragment, it does not\n // have a root element to add a class to. So, we add this attribute to the\n // host element instead:\n scope.host.setAttribute('data-js-focus-visible', '');\n } else if (scope.nodeType === Node.DOCUMENT_NODE) {\n document.documentElement.classList.add('js-focus-visible');\n document.documentElement.setAttribute('data-js-focus-visible', '');\n }\n }\n\n // It is important to wrap all references to global window and document in\n // these checks to support server-side rendering use cases\n // @see https://github.com/WICG/focus-visible/issues/199\n if (typeof window !== 'undefined' && typeof document !== 'undefined') {\n // Make the polyfill helper globally available. This can be used as a signal\n // to interested libraries that wish to coordinate with the polyfill for e.g.,\n // applying the polyfill to a shadow root:\n window.applyFocusVisiblePolyfill = applyFocusVisiblePolyfill;\n\n // Notify interested libraries of the polyfill's presence, in case the\n // polyfill was loaded lazily:\n var event;\n\n try {\n event = new CustomEvent('focus-visible-polyfill-ready');\n } catch (error) {\n // IE11 does not support using CustomEvent as a constructor directly:\n event = document.createEvent('CustomEvent');\n event.initCustomEvent('focus-visible-polyfill-ready', false, false, {});\n }\n\n window.dispatchEvent(event);\n }\n\n if (typeof document !== 'undefined') {\n // Apply the polyfill to the global document, so that no JavaScript\n // coordination is required to use the polyfill in the top-level document:\n applyFocusVisiblePolyfill(document);\n }\n\n})));\n", "/*!\n * escape-html\n * Copyright(c) 2012-2013 TJ Holowaychuk\n * Copyright(c) 2015 Andreas Lubbe\n * Copyright(c) 2015 Tiancheng \"Timothy\" Gu\n * MIT Licensed\n */\n\n'use strict';\n\n/**\n * Module variables.\n * @private\n */\n\nvar matchHtmlRegExp = /[\"'&<>]/;\n\n/**\n * Module exports.\n * @public\n */\n\nmodule.exports = escapeHtml;\n\n/**\n * Escape special characters in the given string of html.\n *\n * @param {string} string The string to escape for inserting into HTML\n * @return {string}\n * @public\n */\n\nfunction escapeHtml(string) {\n var str = '' + string;\n var match = matchHtmlRegExp.exec(str);\n\n if (!match) {\n return str;\n }\n\n var escape;\n var html = '';\n var index = 0;\n var lastIndex = 0;\n\n for (index = match.index; index < str.length; index++) {\n switch (str.charCodeAt(index)) {\n case 34: // \"\n escape = '"';\n break;\n case 38: // &\n escape = '&';\n break;\n case 39: // '\n escape = ''';\n break;\n case 60: // <\n escape = '<';\n break;\n case 62: // >\n escape = '>';\n break;\n default:\n continue;\n }\n\n if (lastIndex !== index) {\n html += str.substring(lastIndex, index);\n }\n\n lastIndex = index + 1;\n html += escape;\n }\n\n return lastIndex !== index\n ? html + str.substring(lastIndex, index)\n : html;\n}\n", "/*!\n * clipboard.js v2.0.11\n * https://clipboardjs.com/\n *\n * Licensed MIT \u00A9 Zeno Rocha\n */\n(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"ClipboardJS\"] = factory();\n\telse\n\t\troot[\"ClipboardJS\"] = factory();\n})(this, function() {\nreturn /******/ (function() { // webpackBootstrap\n/******/ \tvar __webpack_modules__ = ({\n\n/***/ 686:\n/***/ (function(__unused_webpack_module, __webpack_exports__, __webpack_require__) {\n\n\"use strict\";\n\n// EXPORTS\n__webpack_require__.d(__webpack_exports__, {\n \"default\": function() { return /* binding */ clipboard; }\n});\n\n// EXTERNAL MODULE: ./node_modules/tiny-emitter/index.js\nvar tiny_emitter = __webpack_require__(279);\nvar tiny_emitter_default = /*#__PURE__*/__webpack_require__.n(tiny_emitter);\n// EXTERNAL MODULE: ./node_modules/good-listener/src/listen.js\nvar listen = __webpack_require__(370);\nvar listen_default = /*#__PURE__*/__webpack_require__.n(listen);\n// EXTERNAL MODULE: ./node_modules/select/src/select.js\nvar src_select = __webpack_require__(817);\nvar select_default = /*#__PURE__*/__webpack_require__.n(src_select);\n;// CONCATENATED MODULE: ./src/common/command.js\n/**\n * Executes a given operation type.\n * @param {String} type\n * @return {Boolean}\n */\nfunction command(type) {\n try {\n return document.execCommand(type);\n } catch (err) {\n return false;\n }\n}\n;// CONCATENATED MODULE: ./src/actions/cut.js\n\n\n/**\n * Cut action wrapper.\n * @param {String|HTMLElement} target\n * @return {String}\n */\n\nvar ClipboardActionCut = function ClipboardActionCut(target) {\n var selectedText = select_default()(target);\n command('cut');\n return selectedText;\n};\n\n/* harmony default export */ var actions_cut = (ClipboardActionCut);\n;// CONCATENATED MODULE: ./src/common/create-fake-element.js\n/**\n * Creates a fake textarea element with a value.\n * @param {String} value\n * @return {HTMLElement}\n */\nfunction createFakeElement(value) {\n var isRTL = document.documentElement.getAttribute('dir') === 'rtl';\n var fakeElement = document.createElement('textarea'); // Prevent zooming on iOS\n\n fakeElement.style.fontSize = '12pt'; // Reset box model\n\n fakeElement.style.border = '0';\n fakeElement.style.padding = '0';\n fakeElement.style.margin = '0'; // Move element out of screen horizontally\n\n fakeElement.style.position = 'absolute';\n fakeElement.style[isRTL ? 'right' : 'left'] = '-9999px'; // Move element to the same position vertically\n\n var yPosition = window.pageYOffset || document.documentElement.scrollTop;\n fakeElement.style.top = \"\".concat(yPosition, \"px\");\n fakeElement.setAttribute('readonly', '');\n fakeElement.value = value;\n return fakeElement;\n}\n;// CONCATENATED MODULE: ./src/actions/copy.js\n\n\n\n/**\n * Create fake copy action wrapper using a fake element.\n * @param {String} target\n * @param {Object} options\n * @return {String}\n */\n\nvar fakeCopyAction = function fakeCopyAction(value, options) {\n var fakeElement = createFakeElement(value);\n options.container.appendChild(fakeElement);\n var selectedText = select_default()(fakeElement);\n command('copy');\n fakeElement.remove();\n return selectedText;\n};\n/**\n * Copy action wrapper.\n * @param {String|HTMLElement} target\n * @param {Object} options\n * @return {String}\n */\n\n\nvar ClipboardActionCopy = function ClipboardActionCopy(target) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {\n container: document.body\n };\n var selectedText = '';\n\n if (typeof target === 'string') {\n selectedText = fakeCopyAction(target, options);\n } else if (target instanceof HTMLInputElement && !['text', 'search', 'url', 'tel', 'password'].includes(target === null || target === void 0 ? void 0 : target.type)) {\n // If input type doesn't support `setSelectionRange`. Simulate it. https://developer.mozilla.org/en-US/docs/Web/API/HTMLInputElement/setSelectionRange\n selectedText = fakeCopyAction(target.value, options);\n } else {\n selectedText = select_default()(target);\n command('copy');\n }\n\n return selectedText;\n};\n\n/* harmony default export */ var actions_copy = (ClipboardActionCopy);\n;// CONCATENATED MODULE: ./src/actions/default.js\nfunction _typeof(obj) { \"@babel/helpers - typeof\"; if (typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }; } return _typeof(obj); }\n\n\n\n/**\n * Inner function which performs selection from either `text` or `target`\n * properties and then executes copy or cut operations.\n * @param {Object} options\n */\n\nvar ClipboardActionDefault = function ClipboardActionDefault() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n // Defines base properties passed from constructor.\n var _options$action = options.action,\n action = _options$action === void 0 ? 'copy' : _options$action,\n container = options.container,\n target = options.target,\n text = options.text; // Sets the `action` to be performed which can be either 'copy' or 'cut'.\n\n if (action !== 'copy' && action !== 'cut') {\n throw new Error('Invalid \"action\" value, use either \"copy\" or \"cut\"');\n } // Sets the `target` property using an element that will be have its content copied.\n\n\n if (target !== undefined) {\n if (target && _typeof(target) === 'object' && target.nodeType === 1) {\n if (action === 'copy' && target.hasAttribute('disabled')) {\n throw new Error('Invalid \"target\" attribute. Please use \"readonly\" instead of \"disabled\" attribute');\n }\n\n if (action === 'cut' && (target.hasAttribute('readonly') || target.hasAttribute('disabled'))) {\n throw new Error('Invalid \"target\" attribute. You can\\'t cut text from elements with \"readonly\" or \"disabled\" attributes');\n }\n } else {\n throw new Error('Invalid \"target\" value, use a valid Element');\n }\n } // Define selection strategy based on `text` property.\n\n\n if (text) {\n return actions_copy(text, {\n container: container\n });\n } // Defines which selection strategy based on `target` property.\n\n\n if (target) {\n return action === 'cut' ? actions_cut(target) : actions_copy(target, {\n container: container\n });\n }\n};\n\n/* harmony default export */ var actions_default = (ClipboardActionDefault);\n;// CONCATENATED MODULE: ./src/clipboard.js\nfunction clipboard_typeof(obj) { \"@babel/helpers - typeof\"; if (typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\") { clipboard_typeof = function _typeof(obj) { return typeof obj; }; } else { clipboard_typeof = function _typeof(obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }; } return clipboard_typeof(obj); }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function\"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); if (superClass) _setPrototypeOf(subClass, superClass); }\n\nfunction _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }\n\nfunction _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }\n\nfunction _possibleConstructorReturn(self, call) { if (call && (clipboard_typeof(call) === \"object\" || typeof call === \"function\")) { return call; } return _assertThisInitialized(self); }\n\nfunction _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return self; }\n\nfunction _isNativeReflectConstruct() { if (typeof Reflect === \"undefined\" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === \"function\") return true; try { Date.prototype.toString.call(Reflect.construct(Date, [], function () {})); return true; } catch (e) { return false; } }\n\nfunction _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }\n\n\n\n\n\n\n/**\n * Helper function to retrieve attribute value.\n * @param {String} suffix\n * @param {Element} element\n */\n\nfunction getAttributeValue(suffix, element) {\n var attribute = \"data-clipboard-\".concat(suffix);\n\n if (!element.hasAttribute(attribute)) {\n return;\n }\n\n return element.getAttribute(attribute);\n}\n/**\n * Base class which takes one or more elements, adds event listeners to them,\n * and instantiates a new `ClipboardAction` on each click.\n */\n\n\nvar Clipboard = /*#__PURE__*/function (_Emitter) {\n _inherits(Clipboard, _Emitter);\n\n var _super = _createSuper(Clipboard);\n\n /**\n * @param {String|HTMLElement|HTMLCollection|NodeList} trigger\n * @param {Object} options\n */\n function Clipboard(trigger, options) {\n var _this;\n\n _classCallCheck(this, Clipboard);\n\n _this = _super.call(this);\n\n _this.resolveOptions(options);\n\n _this.listenClick(trigger);\n\n return _this;\n }\n /**\n * Defines if attributes would be resolved using internal setter functions\n * or custom functions that were passed in the constructor.\n * @param {Object} options\n */\n\n\n _createClass(Clipboard, [{\n key: \"resolveOptions\",\n value: function resolveOptions() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n this.action = typeof options.action === 'function' ? options.action : this.defaultAction;\n this.target = typeof options.target === 'function' ? options.target : this.defaultTarget;\n this.text = typeof options.text === 'function' ? options.text : this.defaultText;\n this.container = clipboard_typeof(options.container) === 'object' ? options.container : document.body;\n }\n /**\n * Adds a click event listener to the passed trigger.\n * @param {String|HTMLElement|HTMLCollection|NodeList} trigger\n */\n\n }, {\n key: \"listenClick\",\n value: function listenClick(trigger) {\n var _this2 = this;\n\n this.listener = listen_default()(trigger, 'click', function (e) {\n return _this2.onClick(e);\n });\n }\n /**\n * Defines a new `ClipboardAction` on each click event.\n * @param {Event} e\n */\n\n }, {\n key: \"onClick\",\n value: function onClick(e) {\n var trigger = e.delegateTarget || e.currentTarget;\n var action = this.action(trigger) || 'copy';\n var text = actions_default({\n action: action,\n container: this.container,\n target: this.target(trigger),\n text: this.text(trigger)\n }); // Fires an event based on the copy operation result.\n\n this.emit(text ? 'success' : 'error', {\n action: action,\n text: text,\n trigger: trigger,\n clearSelection: function clearSelection() {\n if (trigger) {\n trigger.focus();\n }\n\n window.getSelection().removeAllRanges();\n }\n });\n }\n /**\n * Default `action` lookup function.\n * @param {Element} trigger\n */\n\n }, {\n key: \"defaultAction\",\n value: function defaultAction(trigger) {\n return getAttributeValue('action', trigger);\n }\n /**\n * Default `target` lookup function.\n * @param {Element} trigger\n */\n\n }, {\n key: \"defaultTarget\",\n value: function defaultTarget(trigger) {\n var selector = getAttributeValue('target', trigger);\n\n if (selector) {\n return document.querySelector(selector);\n }\n }\n /**\n * Allow fire programmatically a copy action\n * @param {String|HTMLElement} target\n * @param {Object} options\n * @returns Text copied.\n */\n\n }, {\n key: \"defaultText\",\n\n /**\n * Default `text` lookup function.\n * @param {Element} trigger\n */\n value: function defaultText(trigger) {\n return getAttributeValue('text', trigger);\n }\n /**\n * Destroy lifecycle.\n */\n\n }, {\n key: \"destroy\",\n value: function destroy() {\n this.listener.destroy();\n }\n }], [{\n key: \"copy\",\n value: function copy(target) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {\n container: document.body\n };\n return actions_copy(target, options);\n }\n /**\n * Allow fire programmatically a cut action\n * @param {String|HTMLElement} target\n * @returns Text cutted.\n */\n\n }, {\n key: \"cut\",\n value: function cut(target) {\n return actions_cut(target);\n }\n /**\n * Returns the support of the given action, or all actions if no action is\n * given.\n * @param {String} [action]\n */\n\n }, {\n key: \"isSupported\",\n value: function isSupported() {\n var action = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : ['copy', 'cut'];\n var actions = typeof action === 'string' ? [action] : action;\n var support = !!document.queryCommandSupported;\n actions.forEach(function (action) {\n support = support && !!document.queryCommandSupported(action);\n });\n return support;\n }\n }]);\n\n return Clipboard;\n}((tiny_emitter_default()));\n\n/* harmony default export */ var clipboard = (Clipboard);\n\n/***/ }),\n\n/***/ 828:\n/***/ (function(module) {\n\nvar DOCUMENT_NODE_TYPE = 9;\n\n/**\n * A polyfill for Element.matches()\n */\nif (typeof Element !== 'undefined' && !Element.prototype.matches) {\n var proto = Element.prototype;\n\n proto.matches = proto.matchesSelector ||\n proto.mozMatchesSelector ||\n proto.msMatchesSelector ||\n proto.oMatchesSelector ||\n proto.webkitMatchesSelector;\n}\n\n/**\n * Finds the closest parent that matches a selector.\n *\n * @param {Element} element\n * @param {String} selector\n * @return {Function}\n */\nfunction closest (element, selector) {\n while (element && element.nodeType !== DOCUMENT_NODE_TYPE) {\n if (typeof element.matches === 'function' &&\n element.matches(selector)) {\n return element;\n }\n element = element.parentNode;\n }\n}\n\nmodule.exports = closest;\n\n\n/***/ }),\n\n/***/ 438:\n/***/ (function(module, __unused_webpack_exports, __webpack_require__) {\n\nvar closest = __webpack_require__(828);\n\n/**\n * Delegates event to a selector.\n *\n * @param {Element} element\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @param {Boolean} useCapture\n * @return {Object}\n */\nfunction _delegate(element, selector, type, callback, useCapture) {\n var listenerFn = listener.apply(this, arguments);\n\n element.addEventListener(type, listenerFn, useCapture);\n\n return {\n destroy: function() {\n element.removeEventListener(type, listenerFn, useCapture);\n }\n }\n}\n\n/**\n * Delegates event to a selector.\n *\n * @param {Element|String|Array} [elements]\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @param {Boolean} useCapture\n * @return {Object}\n */\nfunction delegate(elements, selector, type, callback, useCapture) {\n // Handle the regular Element usage\n if (typeof elements.addEventListener === 'function') {\n return _delegate.apply(null, arguments);\n }\n\n // Handle Element-less usage, it defaults to global delegation\n if (typeof type === 'function') {\n // Use `document` as the first parameter, then apply arguments\n // This is a short way to .unshift `arguments` without running into deoptimizations\n return _delegate.bind(null, document).apply(null, arguments);\n }\n\n // Handle Selector-based usage\n if (typeof elements === 'string') {\n elements = document.querySelectorAll(elements);\n }\n\n // Handle Array-like based usage\n return Array.prototype.map.call(elements, function (element) {\n return _delegate(element, selector, type, callback, useCapture);\n });\n}\n\n/**\n * Finds closest match and invokes callback.\n *\n * @param {Element} element\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @return {Function}\n */\nfunction listener(element, selector, type, callback) {\n return function(e) {\n e.delegateTarget = closest(e.target, selector);\n\n if (e.delegateTarget) {\n callback.call(element, e);\n }\n }\n}\n\nmodule.exports = delegate;\n\n\n/***/ }),\n\n/***/ 879:\n/***/ (function(__unused_webpack_module, exports) {\n\n/**\n * Check if argument is a HTML element.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.node = function(value) {\n return value !== undefined\n && value instanceof HTMLElement\n && value.nodeType === 1;\n};\n\n/**\n * Check if argument is a list of HTML elements.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.nodeList = function(value) {\n var type = Object.prototype.toString.call(value);\n\n return value !== undefined\n && (type === '[object NodeList]' || type === '[object HTMLCollection]')\n && ('length' in value)\n && (value.length === 0 || exports.node(value[0]));\n};\n\n/**\n * Check if argument is a string.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.string = function(value) {\n return typeof value === 'string'\n || value instanceof String;\n};\n\n/**\n * Check if argument is a function.\n *\n * @param {Object} value\n * @return {Boolean}\n */\nexports.fn = function(value) {\n var type = Object.prototype.toString.call(value);\n\n return type === '[object Function]';\n};\n\n\n/***/ }),\n\n/***/ 370:\n/***/ (function(module, __unused_webpack_exports, __webpack_require__) {\n\nvar is = __webpack_require__(879);\nvar delegate = __webpack_require__(438);\n\n/**\n * Validates all params and calls the right\n * listener function based on its target type.\n *\n * @param {String|HTMLElement|HTMLCollection|NodeList} target\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listen(target, type, callback) {\n if (!target && !type && !callback) {\n throw new Error('Missing required arguments');\n }\n\n if (!is.string(type)) {\n throw new TypeError('Second argument must be a String');\n }\n\n if (!is.fn(callback)) {\n throw new TypeError('Third argument must be a Function');\n }\n\n if (is.node(target)) {\n return listenNode(target, type, callback);\n }\n else if (is.nodeList(target)) {\n return listenNodeList(target, type, callback);\n }\n else if (is.string(target)) {\n return listenSelector(target, type, callback);\n }\n else {\n throw new TypeError('First argument must be a String, HTMLElement, HTMLCollection, or NodeList');\n }\n}\n\n/**\n * Adds an event listener to a HTML element\n * and returns a remove listener function.\n *\n * @param {HTMLElement} node\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenNode(node, type, callback) {\n node.addEventListener(type, callback);\n\n return {\n destroy: function() {\n node.removeEventListener(type, callback);\n }\n }\n}\n\n/**\n * Add an event listener to a list of HTML elements\n * and returns a remove listener function.\n *\n * @param {NodeList|HTMLCollection} nodeList\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenNodeList(nodeList, type, callback) {\n Array.prototype.forEach.call(nodeList, function(node) {\n node.addEventListener(type, callback);\n });\n\n return {\n destroy: function() {\n Array.prototype.forEach.call(nodeList, function(node) {\n node.removeEventListener(type, callback);\n });\n }\n }\n}\n\n/**\n * Add an event listener to a selector\n * and returns a remove listener function.\n *\n * @param {String} selector\n * @param {String} type\n * @param {Function} callback\n * @return {Object}\n */\nfunction listenSelector(selector, type, callback) {\n return delegate(document.body, selector, type, callback);\n}\n\nmodule.exports = listen;\n\n\n/***/ }),\n\n/***/ 817:\n/***/ (function(module) {\n\nfunction select(element) {\n var selectedText;\n\n if (element.nodeName === 'SELECT') {\n element.focus();\n\n selectedText = element.value;\n }\n else if (element.nodeName === 'INPUT' || element.nodeName === 'TEXTAREA') {\n var isReadOnly = element.hasAttribute('readonly');\n\n if (!isReadOnly) {\n element.setAttribute('readonly', '');\n }\n\n element.select();\n element.setSelectionRange(0, element.value.length);\n\n if (!isReadOnly) {\n element.removeAttribute('readonly');\n }\n\n selectedText = element.value;\n }\n else {\n if (element.hasAttribute('contenteditable')) {\n element.focus();\n }\n\n var selection = window.getSelection();\n var range = document.createRange();\n\n range.selectNodeContents(element);\n selection.removeAllRanges();\n selection.addRange(range);\n\n selectedText = selection.toString();\n }\n\n return selectedText;\n}\n\nmodule.exports = select;\n\n\n/***/ }),\n\n/***/ 279:\n/***/ (function(module) {\n\nfunction E () {\n // Keep this empty so it's easier to inherit from\n // (via https://github.com/lipsmack from https://github.com/scottcorgan/tiny-emitter/issues/3)\n}\n\nE.prototype = {\n on: function (name, callback, ctx) {\n var e = this.e || (this.e = {});\n\n (e[name] || (e[name] = [])).push({\n fn: callback,\n ctx: ctx\n });\n\n return this;\n },\n\n once: function (name, callback, ctx) {\n var self = this;\n function listener () {\n self.off(name, listener);\n callback.apply(ctx, arguments);\n };\n\n listener._ = callback\n return this.on(name, listener, ctx);\n },\n\n emit: function (name) {\n var data = [].slice.call(arguments, 1);\n var evtArr = ((this.e || (this.e = {}))[name] || []).slice();\n var i = 0;\n var len = evtArr.length;\n\n for (i; i < len; i++) {\n evtArr[i].fn.apply(evtArr[i].ctx, data);\n }\n\n return this;\n },\n\n off: function (name, callback) {\n var e = this.e || (this.e = {});\n var evts = e[name];\n var liveEvents = [];\n\n if (evts && callback) {\n for (var i = 0, len = evts.length; i < len; i++) {\n if (evts[i].fn !== callback && evts[i].fn._ !== callback)\n liveEvents.push(evts[i]);\n }\n }\n\n // Remove event from queue to prevent memory leak\n // Suggested by https://github.com/lazd\n // Ref: https://github.com/scottcorgan/tiny-emitter/commit/c6ebfaa9bc973b33d110a84a307742b7cf94c953#commitcomment-5024910\n\n (liveEvents.length)\n ? e[name] = liveEvents\n : delete e[name];\n\n return this;\n }\n};\n\nmodule.exports = E;\nmodule.exports.TinyEmitter = E;\n\n\n/***/ })\n\n/******/ \t});\n/************************************************************************/\n/******/ \t// The module cache\n/******/ \tvar __webpack_module_cache__ = {};\n/******/ \t\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(__webpack_module_cache__[moduleId]) {\n/******/ \t\t\treturn __webpack_module_cache__[moduleId].exports;\n/******/ \t\t}\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = __webpack_module_cache__[moduleId] = {\n/******/ \t\t\t// no module.id needed\n/******/ \t\t\t// no module.loaded needed\n/******/ \t\t\texports: {}\n/******/ \t\t};\n/******/ \t\n/******/ \t\t// Execute the module function\n/******/ \t\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n/******/ \t\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/ \t\n/************************************************************************/\n/******/ \t/* webpack/runtime/compat get default export */\n/******/ \t!function() {\n/******/ \t\t// getDefaultExport function for compatibility with non-harmony modules\n/******/ \t\t__webpack_require__.n = function(module) {\n/******/ \t\t\tvar getter = module && module.__esModule ?\n/******/ \t\t\t\tfunction() { return module['default']; } :\n/******/ \t\t\t\tfunction() { return module; };\n/******/ \t\t\t__webpack_require__.d(getter, { a: getter });\n/******/ \t\t\treturn getter;\n/******/ \t\t};\n/******/ \t}();\n/******/ \t\n/******/ \t/* webpack/runtime/define property getters */\n/******/ \t!function() {\n/******/ \t\t// define getter functions for harmony exports\n/******/ \t\t__webpack_require__.d = function(exports, definition) {\n/******/ \t\t\tfor(var key in definition) {\n/******/ \t\t\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n/******/ \t\t\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n/******/ \t\t\t\t}\n/******/ \t\t\t}\n/******/ \t\t};\n/******/ \t}();\n/******/ \t\n/******/ \t/* webpack/runtime/hasOwnProperty shorthand */\n/******/ \t!function() {\n/******/ \t\t__webpack_require__.o = function(obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }\n/******/ \t}();\n/******/ \t\n/************************************************************************/\n/******/ \t// module exports must be returned from runtime so entry inlining is disabled\n/******/ \t// startup\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(686);\n/******/ })()\n.default;\n});", "/*\n * Copyright (c) 2016-2024 Martin Donath \n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to\n * deal in the Software without restriction, including without limitation the\n * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n * sell copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in\n * all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS\n * IN THE SOFTWARE.\n */\n\nimport \"focus-visible\"\n\nimport {\n EMPTY,\n NEVER,\n Observable,\n Subject,\n defer,\n delay,\n filter,\n map,\n merge,\n mergeWith,\n shareReplay,\n switchMap\n} from \"rxjs\"\n\nimport { configuration, feature } from \"./_\"\nimport {\n at,\n getActiveElement,\n getOptionalElement,\n requestJSON,\n setLocation,\n setToggle,\n watchDocument,\n watchKeyboard,\n watchLocation,\n watchLocationTarget,\n watchMedia,\n watchPrint,\n watchScript,\n watchViewport\n} from \"./browser\"\nimport {\n getComponentElement,\n getComponentElements,\n mountAnnounce,\n mountBackToTop,\n mountConsent,\n mountContent,\n mountDialog,\n mountHeader,\n mountHeaderTitle,\n mountPalette,\n mountProgress,\n mountSearch,\n mountSearchHiglight,\n mountSidebar,\n mountSource,\n mountTableOfContents,\n mountTabs,\n watchHeader,\n watchMain\n} from \"./components\"\nimport {\n SearchIndex,\n setupClipboardJS,\n setupInstantNavigation,\n setupVersionSelector\n} from \"./integrations\"\nimport {\n patchEllipsis,\n patchIndeterminate,\n patchScrollfix,\n patchScrolllock\n} from \"./patches\"\nimport \"./polyfills\"\n\n/* ----------------------------------------------------------------------------\n * Functions - @todo refactor\n * ------------------------------------------------------------------------- */\n\n/**\n * Fetch search index\n *\n * @returns Search index observable\n */\nfunction fetchSearchIndex(): Observable {\n if (location.protocol === \"file:\") {\n return watchScript(\n `${new URL(\"search/search_index.js\", config.base)}`\n )\n .pipe(\n // @ts-ignore - @todo fix typings\n map(() => __index),\n shareReplay(1)\n )\n } else {\n return requestJSON(\n new URL(\"search/search_index.json\", config.base)\n )\n }\n}\n\n/* ----------------------------------------------------------------------------\n * Application\n * ------------------------------------------------------------------------- */\n\n/* Yay, JavaScript is available */\ndocument.documentElement.classList.remove(\"no-js\")\ndocument.documentElement.classList.add(\"js\")\n\n/* Set up navigation observables and subjects */\nconst document$ = watchDocument()\nconst location$ = watchLocation()\nconst target$ = watchLocationTarget(location$)\nconst keyboard$ = watchKeyboard()\n\n/* Set up media observables */\nconst viewport$ = watchViewport()\nconst tablet$ = watchMedia(\"(min-width: 960px)\")\nconst screen$ = watchMedia(\"(min-width: 1220px)\")\nconst print$ = watchPrint()\n\n/* Retrieve search index, if search is enabled */\nconst config = configuration()\nconst index$ = document.forms.namedItem(\"search\")\n ? fetchSearchIndex()\n : NEVER\n\n/* Set up Clipboard.js integration */\nconst alert$ = new Subject()\nsetupClipboardJS({ alert$ })\n\n/* Set up progress indicator */\nconst progress$ = new Subject()\n\n/* Set up instant navigation, if enabled */\nif (feature(\"navigation.instant\"))\n setupInstantNavigation({ location$, viewport$, progress$ })\n .subscribe(document$)\n\n/* Set up version selector */\nif (config.version?.provider === \"mike\")\n setupVersionSelector({ document$ })\n\n/* Always close drawer and search on navigation */\nmerge(location$, target$)\n .pipe(\n delay(125)\n )\n .subscribe(() => {\n setToggle(\"drawer\", false)\n setToggle(\"search\", false)\n })\n\n/* Set up global keyboard handlers */\nkeyboard$\n .pipe(\n filter(({ mode }) => mode === \"global\")\n )\n .subscribe(key => {\n switch (key.type) {\n\n /* Go to previous page */\n case \"p\":\n case \",\":\n const prev = getOptionalElement(\"link[rel=prev]\")\n if (typeof prev !== \"undefined\")\n setLocation(prev)\n break\n\n /* Go to next page */\n case \"n\":\n case \".\":\n const next = getOptionalElement(\"link[rel=next]\")\n if (typeof next !== \"undefined\")\n setLocation(next)\n break\n\n /* Expand navigation, see https://bit.ly/3ZjG5io */\n case \"Enter\":\n const active = getActiveElement()\n if (active instanceof HTMLLabelElement)\n active.click()\n }\n })\n\n/* Set up patches */\npatchEllipsis({ viewport$, document$ })\npatchIndeterminate({ document$, tablet$ })\npatchScrollfix({ document$ })\npatchScrolllock({ viewport$, tablet$ })\n\n/* Set up header and main area observable */\nconst header$ = watchHeader(getComponentElement(\"header\"), { viewport$ })\nconst main$ = document$\n .pipe(\n map(() => getComponentElement(\"main\")),\n switchMap(el => watchMain(el, { viewport$, header$ })),\n shareReplay(1)\n )\n\n/* Set up control component observables */\nconst control$ = merge(\n\n /* Consent */\n ...getComponentElements(\"consent\")\n .map(el => mountConsent(el, { target$ })),\n\n /* Dialog */\n ...getComponentElements(\"dialog\")\n .map(el => mountDialog(el, { alert$ })),\n\n /* Color palette */\n ...getComponentElements(\"palette\")\n .map(el => mountPalette(el)),\n\n /* Progress bar */\n ...getComponentElements(\"progress\")\n .map(el => mountProgress(el, { progress$ })),\n\n /* Search */\n ...getComponentElements(\"search\")\n .map(el => mountSearch(el, { index$, keyboard$ })),\n\n /* Repository information */\n ...getComponentElements(\"source\")\n .map(el => mountSource(el))\n)\n\n/* Set up content component observables */\nconst content$ = defer(() => merge(\n\n /* Announcement bar */\n ...getComponentElements(\"announce\")\n .map(el => mountAnnounce(el)),\n\n /* Content */\n ...getComponentElements(\"content\")\n .map(el => mountContent(el, { viewport$, target$, print$ })),\n\n /* Search highlighting */\n ...getComponentElements(\"content\")\n .map(el => feature(\"search.highlight\")\n ? mountSearchHiglight(el, { index$, location$ })\n : EMPTY\n ),\n\n /* Header */\n ...getComponentElements(\"header\")\n .map(el => mountHeader(el, { viewport$, header$, main$ })),\n\n /* Header title */\n ...getComponentElements(\"header-title\")\n .map(el => mountHeaderTitle(el, { viewport$, header$ })),\n\n /* Sidebar */\n ...getComponentElements(\"sidebar\")\n .map(el => el.getAttribute(\"data-md-type\") === \"navigation\"\n ? at(screen$, () => mountSidebar(el, { viewport$, header$, main$ }))\n : at(tablet$, () => mountSidebar(el, { viewport$, header$, main$ }))\n ),\n\n /* Navigation tabs */\n ...getComponentElements(\"tabs\")\n .map(el => mountTabs(el, { viewport$, header$ })),\n\n /* Table of contents */\n ...getComponentElements(\"toc\")\n .map(el => mountTableOfContents(el, {\n viewport$, header$, main$, target$\n })),\n\n /* Back-to-top button */\n ...getComponentElements(\"top\")\n .map(el => mountBackToTop(el, { viewport$, header$, main$, target$ }))\n))\n\n/* Set up component observables */\nconst component$ = document$\n .pipe(\n switchMap(() => content$),\n mergeWith(control$),\n shareReplay(1)\n )\n\n/* Subscribe to all components */\ncomponent$.subscribe()\n\n/* ----------------------------------------------------------------------------\n * Exports\n * ------------------------------------------------------------------------- */\n\nwindow.document$ = document$ /* Document observable */\nwindow.location$ = location$ /* Location subject */\nwindow.target$ = target$ /* Location target observable */\nwindow.keyboard$ = keyboard$ /* Keyboard observable */\nwindow.viewport$ = viewport$ /* Viewport observable */\nwindow.tablet$ = tablet$ /* Media tablet observable */\nwindow.screen$ = screen$ /* Media screen observable */\nwindow.print$ = print$ /* Media print observable */\nwindow.alert$ = alert$ /* Alert subject */\nwindow.progress$ = progress$ /* Progress indicator subject */\nwindow.component$ = component$ /* Component observable */\n", "/******************************************************************************\nCopyright (c) Microsoft Corporation.\n\nPermission to use, copy, modify, and/or distribute this software for any\npurpose with or without fee is hereby granted.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\nPERFORMANCE OF THIS SOFTWARE.\n***************************************************************************** */\n/* global Reflect, Promise, SuppressedError, Symbol, Iterator */\n\nvar extendStatics = function(d, b) {\n extendStatics = Object.setPrototypeOf ||\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\n return extendStatics(d, b);\n};\n\nexport function __extends(d, b) {\n if (typeof b !== \"function\" && b !== null)\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\n extendStatics(d, b);\n function __() { this.constructor = d; }\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\n}\n\nexport var __assign = function() {\n __assign = Object.assign || function __assign(t) {\n for (var s, i = 1, n = arguments.length; i < n; i++) {\n s = arguments[i];\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\n }\n return t;\n }\n return __assign.apply(this, arguments);\n}\n\nexport function __rest(s, e) {\n var t = {};\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\n t[p] = s[p];\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\n t[p[i]] = s[p[i]];\n }\n return t;\n}\n\nexport function __decorate(decorators, target, key, desc) {\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\n return c > 3 && r && Object.defineProperty(target, key, r), r;\n}\n\nexport function __param(paramIndex, decorator) {\n return function (target, key) { decorator(target, key, paramIndex); }\n}\n\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\n var _, done = false;\n for (var i = decorators.length - 1; i >= 0; i--) {\n var context = {};\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\n if (kind === \"accessor\") {\n if (result === void 0) continue;\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\n if (_ = accept(result.get)) descriptor.get = _;\n if (_ = accept(result.set)) descriptor.set = _;\n if (_ = accept(result.init)) initializers.unshift(_);\n }\n else if (_ = accept(result)) {\n if (kind === \"field\") initializers.unshift(_);\n else descriptor[key] = _;\n }\n }\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\n done = true;\n};\n\nexport function __runInitializers(thisArg, initializers, value) {\n var useValue = arguments.length > 2;\n for (var i = 0; i < initializers.length; i++) {\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\n }\n return useValue ? value : void 0;\n};\n\nexport function __propKey(x) {\n return typeof x === \"symbol\" ? x : \"\".concat(x);\n};\n\nexport function __setFunctionName(f, name, prefix) {\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\n};\n\nexport function __metadata(metadataKey, metadataValue) {\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\n}\n\nexport function __awaiter(thisArg, _arguments, P, generator) {\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n}\n\nexport function __generator(thisArg, body) {\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === \"function\" ? Iterator : Object).prototype);\n return g.next = verb(0), g[\"throw\"] = verb(1), g[\"return\"] = verb(2), typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n function verb(n) { return function (v) { return step([n, v]); }; }\n function step(op) {\n if (f) throw new TypeError(\"Generator is already executing.\");\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n if (y = 0, t) op = [op[0] & 2, t.value];\n switch (op[0]) {\n case 0: case 1: t = op; break;\n case 4: _.label++; return { value: op[1], done: false };\n case 5: _.label++; y = op[1]; op = [0]; continue;\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\n default:\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n if (t[2]) _.ops.pop();\n _.trys.pop(); continue;\n }\n op = body.call(thisArg, _);\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n }\n}\n\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n var desc = Object.getOwnPropertyDescriptor(m, k);\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\n desc = { enumerable: true, get: function() { return m[k]; } };\n }\n Object.defineProperty(o, k2, desc);\n}) : (function(o, m, k, k2) {\n if (k2 === undefined) k2 = k;\n o[k2] = m[k];\n});\n\nexport function __exportStar(m, o) {\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\n}\n\nexport function __values(o) {\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\n if (m) return m.call(o);\n if (o && typeof o.length === \"number\") return {\n next: function () {\n if (o && i >= o.length) o = void 0;\n return { value: o && o[i++], done: !o };\n }\n };\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\n}\n\nexport function __read(o, n) {\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\n if (!m) return o;\n var i = m.call(o), r, ar = [], e;\n try {\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\n }\n catch (error) { e = { error: error }; }\n finally {\n try {\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\n }\n finally { if (e) throw e.error; }\n }\n return ar;\n}\n\n/** @deprecated */\nexport function __spread() {\n for (var ar = [], i = 0; i < arguments.length; i++)\n ar = ar.concat(__read(arguments[i]));\n return ar;\n}\n\n/** @deprecated */\nexport function __spreadArrays() {\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n r[k] = a[j];\n return r;\n}\n\nexport function __spreadArray(to, from, pack) {\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\n if (ar || !(i in from)) {\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\n ar[i] = from[i];\n }\n }\n return to.concat(ar || Array.prototype.slice.call(from));\n}\n\nexport function __await(v) {\n return this instanceof __await ? (this.v = v, this) : new __await(v);\n}\n\nexport function __asyncGenerator(thisArg, _arguments, generator) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\n return i = Object.create((typeof AsyncIterator === \"function\" ? AsyncIterator : Object).prototype), verb(\"next\"), verb(\"throw\"), verb(\"return\", awaitReturn), i[Symbol.asyncIterator] = function () { return this; }, i;\n function awaitReturn(f) { return function (v) { return Promise.resolve(v).then(f, reject); }; }\n function verb(n, f) { if (g[n]) { i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; if (f) i[n] = f(i[n]); } }\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\n function fulfill(value) { resume(\"next\", value); }\n function reject(value) { resume(\"throw\", value); }\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\n}\n\nexport function __asyncDelegator(o) {\n var i, p;\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\n}\n\nexport function __asyncValues(o) {\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\n var m = o[Symbol.asyncIterator], i;\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\n}\n\nexport function __makeTemplateObject(cooked, raw) {\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\n return cooked;\n};\n\nvar __setModuleDefault = Object.create ? (function(o, v) {\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\n}) : function(o, v) {\n o[\"default\"] = v;\n};\n\nexport function __importStar(mod) {\n if (mod && mod.__esModule) return mod;\n var result = {};\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\n __setModuleDefault(result, mod);\n return result;\n}\n\nexport function __importDefault(mod) {\n return (mod && mod.__esModule) ? mod : { default: mod };\n}\n\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\n}\n\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\n}\n\nexport function __classPrivateFieldIn(state, receiver) {\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\n}\n\nexport function __addDisposableResource(env, value, async) {\n if (value !== null && value !== void 0) {\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\n var dispose, inner;\n if (async) {\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\n dispose = value[Symbol.asyncDispose];\n }\n if (dispose === void 0) {\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\n dispose = value[Symbol.dispose];\n if (async) inner = dispose;\n }\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\n if (inner) dispose = function() { try { inner.call(this); } catch (e) { return Promise.reject(e); } };\n env.stack.push({ value: value, dispose: dispose, async: async });\n }\n else if (async) {\n env.stack.push({ async: true });\n }\n return value;\n}\n\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\n var e = new Error(message);\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\n};\n\nexport function __disposeResources(env) {\n function fail(e) {\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\n env.hasError = true;\n }\n var r, s = 0;\n function next() {\n while (r = env.stack.pop()) {\n try {\n if (!r.async && s === 1) return s = 0, env.stack.push(r), Promise.resolve().then(next);\n if (r.dispose) {\n var result = r.dispose.call(r.value);\n if (r.async) return s |= 2, Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\n }\n else s |= 1;\n }\n catch (e) {\n fail(e);\n }\n }\n if (s === 1) return env.hasError ? Promise.reject(env.error) : Promise.resolve();\n if (env.hasError) throw env.error;\n }\n return next();\n}\n\nexport default {\n __extends,\n __assign,\n __rest,\n __decorate,\n __param,\n __metadata,\n __awaiter,\n __generator,\n __createBinding,\n __exportStar,\n __values,\n __read,\n __spread,\n __spreadArrays,\n __spreadArray,\n __await,\n __asyncGenerator,\n __asyncDelegator,\n __asyncValues,\n __makeTemplateObject,\n __importStar,\n __importDefault,\n __classPrivateFieldGet,\n __classPrivateFieldSet,\n __classPrivateFieldIn,\n __addDisposableResource,\n __disposeResources,\n};\n", "/**\n * Returns true if the object is a function.\n * @param value The value to check\n */\nexport function isFunction(value: any): value is (...args: any[]) => any {\n return typeof value === 'function';\n}\n", "/**\n * Used to create Error subclasses until the community moves away from ES5.\n *\n * This is because compiling from TypeScript down to ES5 has issues with subclassing Errors\n * as well as other built-in types: https://github.com/Microsoft/TypeScript/issues/12123\n *\n * @param createImpl A factory function to create the actual constructor implementation. The returned\n * function should be a named function that calls `_super` internally.\n */\nexport function createErrorClass(createImpl: (_super: any) => any): T {\n const _super = (instance: any) => {\n Error.call(instance);\n instance.stack = new Error().stack;\n };\n\n const ctorFunc = createImpl(_super);\n ctorFunc.prototype = Object.create(Error.prototype);\n ctorFunc.prototype.constructor = ctorFunc;\n return ctorFunc;\n}\n", "import { createErrorClass } from './createErrorClass';\n\nexport interface UnsubscriptionError extends Error {\n readonly errors: any[];\n}\n\nexport interface UnsubscriptionErrorCtor {\n /**\n * @deprecated Internal implementation detail. Do not construct error instances.\n * Cannot be tagged as internal: https://github.com/ReactiveX/rxjs/issues/6269\n */\n new (errors: any[]): UnsubscriptionError;\n}\n\n/**\n * An error thrown when one or more errors have occurred during the\n * `unsubscribe` of a {@link Subscription}.\n */\nexport const UnsubscriptionError: UnsubscriptionErrorCtor = createErrorClass(\n (_super) =>\n function UnsubscriptionErrorImpl(this: any, errors: (Error | string)[]) {\n _super(this);\n this.message = errors\n ? `${errors.length} errors occurred during unsubscription:\n${errors.map((err, i) => `${i + 1}) ${err.toString()}`).join('\\n ')}`\n : '';\n this.name = 'UnsubscriptionError';\n this.errors = errors;\n }\n);\n", "/**\n * Removes an item from an array, mutating it.\n * @param arr The array to remove the item from\n * @param item The item to remove\n */\nexport function arrRemove(arr: T[] | undefined | null, item: T) {\n if (arr) {\n const index = arr.indexOf(item);\n 0 <= index && arr.splice(index, 1);\n }\n}\n", "import { isFunction } from './util/isFunction';\nimport { UnsubscriptionError } from './util/UnsubscriptionError';\nimport { SubscriptionLike, TeardownLogic, Unsubscribable } from './types';\nimport { arrRemove } from './util/arrRemove';\n\n/**\n * Represents a disposable resource, such as the execution of an Observable. A\n * Subscription has one important method, `unsubscribe`, that takes no argument\n * and just disposes the resource held by the subscription.\n *\n * Additionally, subscriptions may be grouped together through the `add()`\n * method, which will attach a child Subscription to the current Subscription.\n * When a Subscription is unsubscribed, all its children (and its grandchildren)\n * will be unsubscribed as well.\n *\n * @class Subscription\n */\nexport class Subscription implements SubscriptionLike {\n /** @nocollapse */\n public static EMPTY = (() => {\n const empty = new Subscription();\n empty.closed = true;\n return empty;\n })();\n\n /**\n * A flag to indicate whether this Subscription has already been unsubscribed.\n */\n public closed = false;\n\n private _parentage: Subscription[] | Subscription | null = null;\n\n /**\n * The list of registered finalizers to execute upon unsubscription. Adding and removing from this\n * list occurs in the {@link #add} and {@link #remove} methods.\n */\n private _finalizers: Exclude[] | null = null;\n\n /**\n * @param initialTeardown A function executed first as part of the finalization\n * process that is kicked off when {@link #unsubscribe} is called.\n */\n constructor(private initialTeardown?: () => void) {}\n\n /**\n * Disposes the resources held by the subscription. May, for instance, cancel\n * an ongoing Observable execution or cancel any other type of work that\n * started when the Subscription was created.\n * @return {void}\n */\n unsubscribe(): void {\n let errors: any[] | undefined;\n\n if (!this.closed) {\n this.closed = true;\n\n // Remove this from it's parents.\n const { _parentage } = this;\n if (_parentage) {\n this._parentage = null;\n if (Array.isArray(_parentage)) {\n for (const parent of _parentage) {\n parent.remove(this);\n }\n } else {\n _parentage.remove(this);\n }\n }\n\n const { initialTeardown: initialFinalizer } = this;\n if (isFunction(initialFinalizer)) {\n try {\n initialFinalizer();\n } catch (e) {\n errors = e instanceof UnsubscriptionError ? e.errors : [e];\n }\n }\n\n const { _finalizers } = this;\n if (_finalizers) {\n this._finalizers = null;\n for (const finalizer of _finalizers) {\n try {\n execFinalizer(finalizer);\n } catch (err) {\n errors = errors ?? [];\n if (err instanceof UnsubscriptionError) {\n errors = [...errors, ...err.errors];\n } else {\n errors.push(err);\n }\n }\n }\n }\n\n if (errors) {\n throw new UnsubscriptionError(errors);\n }\n }\n }\n\n /**\n * Adds a finalizer to this subscription, so that finalization will be unsubscribed/called\n * when this subscription is unsubscribed. If this subscription is already {@link #closed},\n * because it has already been unsubscribed, then whatever finalizer is passed to it\n * will automatically be executed (unless the finalizer itself is also a closed subscription).\n *\n * Closed Subscriptions cannot be added as finalizers to any subscription. Adding a closed\n * subscription to a any subscription will result in no operation. (A noop).\n *\n * Adding a subscription to itself, or adding `null` or `undefined` will not perform any\n * operation at all. (A noop).\n *\n * `Subscription` instances that are added to this instance will automatically remove themselves\n * if they are unsubscribed. Functions and {@link Unsubscribable} objects that you wish to remove\n * will need to be removed manually with {@link #remove}\n *\n * @param teardown The finalization logic to add to this subscription.\n */\n add(teardown: TeardownLogic): void {\n // Only add the finalizer if it's not undefined\n // and don't add a subscription to itself.\n if (teardown && teardown !== this) {\n if (this.closed) {\n // If this subscription is already closed,\n // execute whatever finalizer is handed to it automatically.\n execFinalizer(teardown);\n } else {\n if (teardown instanceof Subscription) {\n // We don't add closed subscriptions, and we don't add the same subscription\n // twice. Subscription unsubscribe is idempotent.\n if (teardown.closed || teardown._hasParent(this)) {\n return;\n }\n teardown._addParent(this);\n }\n (this._finalizers = this._finalizers ?? []).push(teardown);\n }\n }\n }\n\n /**\n * Checks to see if a this subscription already has a particular parent.\n * This will signal that this subscription has already been added to the parent in question.\n * @param parent the parent to check for\n */\n private _hasParent(parent: Subscription) {\n const { _parentage } = this;\n return _parentage === parent || (Array.isArray(_parentage) && _parentage.includes(parent));\n }\n\n /**\n * Adds a parent to this subscription so it can be removed from the parent if it\n * unsubscribes on it's own.\n *\n * NOTE: THIS ASSUMES THAT {@link _hasParent} HAS ALREADY BEEN CHECKED.\n * @param parent The parent subscription to add\n */\n private _addParent(parent: Subscription) {\n const { _parentage } = this;\n this._parentage = Array.isArray(_parentage) ? (_parentage.push(parent), _parentage) : _parentage ? [_parentage, parent] : parent;\n }\n\n /**\n * Called on a child when it is removed via {@link #remove}.\n * @param parent The parent to remove\n */\n private _removeParent(parent: Subscription) {\n const { _parentage } = this;\n if (_parentage === parent) {\n this._parentage = null;\n } else if (Array.isArray(_parentage)) {\n arrRemove(_parentage, parent);\n }\n }\n\n /**\n * Removes a finalizer from this subscription that was previously added with the {@link #add} method.\n *\n * Note that `Subscription` instances, when unsubscribed, will automatically remove themselves\n * from every other `Subscription` they have been added to. This means that using the `remove` method\n * is not a common thing and should be used thoughtfully.\n *\n * If you add the same finalizer instance of a function or an unsubscribable object to a `Subscription` instance\n * more than once, you will need to call `remove` the same number of times to remove all instances.\n *\n * All finalizer instances are removed to free up memory upon unsubscription.\n *\n * @param teardown The finalizer to remove from this subscription\n */\n remove(teardown: Exclude): void {\n const { _finalizers } = this;\n _finalizers && arrRemove(_finalizers, teardown);\n\n if (teardown instanceof Subscription) {\n teardown._removeParent(this);\n }\n }\n}\n\nexport const EMPTY_SUBSCRIPTION = Subscription.EMPTY;\n\nexport function isSubscription(value: any): value is Subscription {\n return (\n value instanceof Subscription ||\n (value && 'closed' in value && isFunction(value.remove) && isFunction(value.add) && isFunction(value.unsubscribe))\n );\n}\n\nfunction execFinalizer(finalizer: Unsubscribable | (() => void)) {\n if (isFunction(finalizer)) {\n finalizer();\n } else {\n finalizer.unsubscribe();\n }\n}\n", "import { Subscriber } from './Subscriber';\nimport { ObservableNotification } from './types';\n\n/**\n * The {@link GlobalConfig} object for RxJS. It is used to configure things\n * like how to react on unhandled errors.\n */\nexport const config: GlobalConfig = {\n onUnhandledError: null,\n onStoppedNotification: null,\n Promise: undefined,\n useDeprecatedSynchronousErrorHandling: false,\n useDeprecatedNextContext: false,\n};\n\n/**\n * The global configuration object for RxJS, used to configure things\n * like how to react on unhandled errors. Accessible via {@link config}\n * object.\n */\nexport interface GlobalConfig {\n /**\n * A registration point for unhandled errors from RxJS. These are errors that\n * cannot were not handled by consuming code in the usual subscription path. For\n * example, if you have this configured, and you subscribe to an observable without\n * providing an error handler, errors from that subscription will end up here. This\n * will _always_ be called asynchronously on another job in the runtime. This is because\n * we do not want errors thrown in this user-configured handler to interfere with the\n * behavior of the library.\n */\n onUnhandledError: ((err: any) => void) | null;\n\n /**\n * A registration point for notifications that cannot be sent to subscribers because they\n * have completed, errored or have been explicitly unsubscribed. By default, next, complete\n * and error notifications sent to stopped subscribers are noops. However, sometimes callers\n * might want a different behavior. For example, with sources that attempt to report errors\n * to stopped subscribers, a caller can configure RxJS to throw an unhandled error instead.\n * This will _always_ be called asynchronously on another job in the runtime. This is because\n * we do not want errors thrown in this user-configured handler to interfere with the\n * behavior of the library.\n */\n onStoppedNotification: ((notification: ObservableNotification, subscriber: Subscriber) => void) | null;\n\n /**\n * The promise constructor used by default for {@link Observable#toPromise toPromise} and {@link Observable#forEach forEach}\n * methods.\n *\n * @deprecated As of version 8, RxJS will no longer support this sort of injection of a\n * Promise constructor. If you need a Promise implementation other than native promises,\n * please polyfill/patch Promise as you see appropriate. Will be removed in v8.\n */\n Promise?: PromiseConstructorLike;\n\n /**\n * If true, turns on synchronous error rethrowing, which is a deprecated behavior\n * in v6 and higher. This behavior enables bad patterns like wrapping a subscribe\n * call in a try/catch block. It also enables producer interference, a nasty bug\n * where a multicast can be broken for all observers by a downstream consumer with\n * an unhandled error. DO NOT USE THIS FLAG UNLESS IT'S NEEDED TO BUY TIME\n * FOR MIGRATION REASONS.\n *\n * @deprecated As of version 8, RxJS will no longer support synchronous throwing\n * of unhandled errors. All errors will be thrown on a separate call stack to prevent bad\n * behaviors described above. Will be removed in v8.\n */\n useDeprecatedSynchronousErrorHandling: boolean;\n\n /**\n * If true, enables an as-of-yet undocumented feature from v5: The ability to access\n * `unsubscribe()` via `this` context in `next` functions created in observers passed\n * to `subscribe`.\n *\n * This is being removed because the performance was severely problematic, and it could also cause\n * issues when types other than POJOs are passed to subscribe as subscribers, as they will likely have\n * their `this` context overwritten.\n *\n * @deprecated As of version 8, RxJS will no longer support altering the\n * context of next functions provided as part of an observer to Subscribe. Instead,\n * you will have access to a subscription or a signal or token that will allow you to do things like\n * unsubscribe and test closed status. Will be removed in v8.\n */\n useDeprecatedNextContext: boolean;\n}\n", "import type { TimerHandle } from './timerHandle';\ntype SetTimeoutFunction = (handler: () => void, timeout?: number, ...args: any[]) => TimerHandle;\ntype ClearTimeoutFunction = (handle: TimerHandle) => void;\n\ninterface TimeoutProvider {\n setTimeout: SetTimeoutFunction;\n clearTimeout: ClearTimeoutFunction;\n delegate:\n | {\n setTimeout: SetTimeoutFunction;\n clearTimeout: ClearTimeoutFunction;\n }\n | undefined;\n}\n\nexport const timeoutProvider: TimeoutProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n setTimeout(handler: () => void, timeout?: number, ...args) {\n const { delegate } = timeoutProvider;\n if (delegate?.setTimeout) {\n return delegate.setTimeout(handler, timeout, ...args);\n }\n return setTimeout(handler, timeout, ...args);\n },\n clearTimeout(handle) {\n const { delegate } = timeoutProvider;\n return (delegate?.clearTimeout || clearTimeout)(handle as any);\n },\n delegate: undefined,\n};\n", "import { config } from '../config';\nimport { timeoutProvider } from '../scheduler/timeoutProvider';\n\n/**\n * Handles an error on another job either with the user-configured {@link onUnhandledError},\n * or by throwing it on that new job so it can be picked up by `window.onerror`, `process.on('error')`, etc.\n *\n * This should be called whenever there is an error that is out-of-band with the subscription\n * or when an error hits a terminal boundary of the subscription and no error handler was provided.\n *\n * @param err the error to report\n */\nexport function reportUnhandledError(err: any) {\n timeoutProvider.setTimeout(() => {\n const { onUnhandledError } = config;\n if (onUnhandledError) {\n // Execute the user-configured error handler.\n onUnhandledError(err);\n } else {\n // Throw so it is picked up by the runtime's uncaught error mechanism.\n throw err;\n }\n });\n}\n", "/* tslint:disable:no-empty */\nexport function noop() { }\n", "import { CompleteNotification, NextNotification, ErrorNotification } from './types';\n\n/**\n * A completion object optimized for memory use and created to be the\n * same \"shape\" as other notifications in v8.\n * @internal\n */\nexport const COMPLETE_NOTIFICATION = (() => createNotification('C', undefined, undefined) as CompleteNotification)();\n\n/**\n * Internal use only. Creates an optimized error notification that is the same \"shape\"\n * as other notifications.\n * @internal\n */\nexport function errorNotification(error: any): ErrorNotification {\n return createNotification('E', undefined, error) as any;\n}\n\n/**\n * Internal use only. Creates an optimized next notification that is the same \"shape\"\n * as other notifications.\n * @internal\n */\nexport function nextNotification(value: T) {\n return createNotification('N', value, undefined) as NextNotification;\n}\n\n/**\n * Ensures that all notifications created internally have the same \"shape\" in v8.\n *\n * TODO: This is only exported to support a crazy legacy test in `groupBy`.\n * @internal\n */\nexport function createNotification(kind: 'N' | 'E' | 'C', value: any, error: any) {\n return {\n kind,\n value,\n error,\n };\n}\n", "import { config } from '../config';\n\nlet context: { errorThrown: boolean; error: any } | null = null;\n\n/**\n * Handles dealing with errors for super-gross mode. Creates a context, in which\n * any synchronously thrown errors will be passed to {@link captureError}. Which\n * will record the error such that it will be rethrown after the call back is complete.\n * TODO: Remove in v8\n * @param cb An immediately executed function.\n */\nexport function errorContext(cb: () => void) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n const isRoot = !context;\n if (isRoot) {\n context = { errorThrown: false, error: null };\n }\n cb();\n if (isRoot) {\n const { errorThrown, error } = context!;\n context = null;\n if (errorThrown) {\n throw error;\n }\n }\n } else {\n // This is the general non-deprecated path for everyone that\n // isn't crazy enough to use super-gross mode (useDeprecatedSynchronousErrorHandling)\n cb();\n }\n}\n\n/**\n * Captures errors only in super-gross mode.\n * @param err the error to capture\n */\nexport function captureError(err: any) {\n if (config.useDeprecatedSynchronousErrorHandling && context) {\n context.errorThrown = true;\n context.error = err;\n }\n}\n", "import { isFunction } from './util/isFunction';\nimport { Observer, ObservableNotification } from './types';\nimport { isSubscription, Subscription } from './Subscription';\nimport { config } from './config';\nimport { reportUnhandledError } from './util/reportUnhandledError';\nimport { noop } from './util/noop';\nimport { nextNotification, errorNotification, COMPLETE_NOTIFICATION } from './NotificationFactories';\nimport { timeoutProvider } from './scheduler/timeoutProvider';\nimport { captureError } from './util/errorContext';\n\n/**\n * Implements the {@link Observer} interface and extends the\n * {@link Subscription} class. While the {@link Observer} is the public API for\n * consuming the values of an {@link Observable}, all Observers get converted to\n * a Subscriber, in order to provide Subscription-like capabilities such as\n * `unsubscribe`. Subscriber is a common type in RxJS, and crucial for\n * implementing operators, but it is rarely used as a public API.\n *\n * @class Subscriber\n */\nexport class Subscriber extends Subscription implements Observer {\n /**\n * A static factory for a Subscriber, given a (potentially partial) definition\n * of an Observer.\n * @param next The `next` callback of an Observer.\n * @param error The `error` callback of an\n * Observer.\n * @param complete The `complete` callback of an\n * Observer.\n * @return A Subscriber wrapping the (partially defined)\n * Observer represented by the given arguments.\n * @nocollapse\n * @deprecated Do not use. Will be removed in v8. There is no replacement for this\n * method, and there is no reason to be creating instances of `Subscriber` directly.\n * If you have a specific use case, please file an issue.\n */\n static create(next?: (x?: T) => void, error?: (e?: any) => void, complete?: () => void): Subscriber {\n return new SafeSubscriber(next, error, complete);\n }\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n protected isStopped: boolean = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n protected destination: Subscriber | Observer; // this `any` is the escape hatch to erase extra type param (e.g. R)\n\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n * There is no reason to directly create an instance of Subscriber. This type is exported for typings reasons.\n */\n constructor(destination?: Subscriber | Observer) {\n super();\n if (destination) {\n this.destination = destination;\n // Automatically chain subscriptions together here.\n // if destination is a Subscription, then it is a Subscriber.\n if (isSubscription(destination)) {\n destination.add(this);\n }\n } else {\n this.destination = EMPTY_OBSERVER;\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `next` from\n * the Observable, with a value. The Observable may call this method 0 or more\n * times.\n * @param {T} [value] The `next` value.\n * @return {void}\n */\n next(value?: T): void {\n if (this.isStopped) {\n handleStoppedNotification(nextNotification(value), this);\n } else {\n this._next(value!);\n }\n }\n\n /**\n * The {@link Observer} callback to receive notifications of type `error` from\n * the Observable, with an attached `Error`. Notifies the Observer that\n * the Observable has experienced an error condition.\n * @param {any} [err] The `error` exception.\n * @return {void}\n */\n error(err?: any): void {\n if (this.isStopped) {\n handleStoppedNotification(errorNotification(err), this);\n } else {\n this.isStopped = true;\n this._error(err);\n }\n }\n\n /**\n * The {@link Observer} callback to receive a valueless notification of type\n * `complete` from the Observable. Notifies the Observer that the Observable\n * has finished sending push-based notifications.\n * @return {void}\n */\n complete(): void {\n if (this.isStopped) {\n handleStoppedNotification(COMPLETE_NOTIFICATION, this);\n } else {\n this.isStopped = true;\n this._complete();\n }\n }\n\n unsubscribe(): void {\n if (!this.closed) {\n this.isStopped = true;\n super.unsubscribe();\n this.destination = null!;\n }\n }\n\n protected _next(value: T): void {\n this.destination.next(value);\n }\n\n protected _error(err: any): void {\n try {\n this.destination.error(err);\n } finally {\n this.unsubscribe();\n }\n }\n\n protected _complete(): void {\n try {\n this.destination.complete();\n } finally {\n this.unsubscribe();\n }\n }\n}\n\n/**\n * This bind is captured here because we want to be able to have\n * compatibility with monoid libraries that tend to use a method named\n * `bind`. In particular, a library called Monio requires this.\n */\nconst _bind = Function.prototype.bind;\n\nfunction bind any>(fn: Fn, thisArg: any): Fn {\n return _bind.call(fn, thisArg);\n}\n\n/**\n * Internal optimization only, DO NOT EXPOSE.\n * @internal\n */\nclass ConsumerObserver implements Observer {\n constructor(private partialObserver: Partial>) {}\n\n next(value: T): void {\n const { partialObserver } = this;\n if (partialObserver.next) {\n try {\n partialObserver.next(value);\n } catch (error) {\n handleUnhandledError(error);\n }\n }\n }\n\n error(err: any): void {\n const { partialObserver } = this;\n if (partialObserver.error) {\n try {\n partialObserver.error(err);\n } catch (error) {\n handleUnhandledError(error);\n }\n } else {\n handleUnhandledError(err);\n }\n }\n\n complete(): void {\n const { partialObserver } = this;\n if (partialObserver.complete) {\n try {\n partialObserver.complete();\n } catch (error) {\n handleUnhandledError(error);\n }\n }\n }\n}\n\nexport class SafeSubscriber extends Subscriber {\n constructor(\n observerOrNext?: Partial> | ((value: T) => void) | null,\n error?: ((e?: any) => void) | null,\n complete?: (() => void) | null\n ) {\n super();\n\n let partialObserver: Partial>;\n if (isFunction(observerOrNext) || !observerOrNext) {\n // The first argument is a function, not an observer. The next\n // two arguments *could* be observers, or they could be empty.\n partialObserver = {\n next: (observerOrNext ?? undefined) as (((value: T) => void) | undefined),\n error: error ?? undefined,\n complete: complete ?? undefined,\n };\n } else {\n // The first argument is a partial observer.\n let context: any;\n if (this && config.useDeprecatedNextContext) {\n // This is a deprecated path that made `this.unsubscribe()` available in\n // next handler functions passed to subscribe. This only exists behind a flag\n // now, as it is *very* slow.\n context = Object.create(observerOrNext);\n context.unsubscribe = () => this.unsubscribe();\n partialObserver = {\n next: observerOrNext.next && bind(observerOrNext.next, context),\n error: observerOrNext.error && bind(observerOrNext.error, context),\n complete: observerOrNext.complete && bind(observerOrNext.complete, context),\n };\n } else {\n // The \"normal\" path. Just use the partial observer directly.\n partialObserver = observerOrNext;\n }\n }\n\n // Wrap the partial observer to ensure it's a full observer, and\n // make sure proper error handling is accounted for.\n this.destination = new ConsumerObserver(partialObserver);\n }\n}\n\nfunction handleUnhandledError(error: any) {\n if (config.useDeprecatedSynchronousErrorHandling) {\n captureError(error);\n } else {\n // Ideal path, we report this as an unhandled error,\n // which is thrown on a new call stack.\n reportUnhandledError(error);\n }\n}\n\n/**\n * An error handler used when no error handler was supplied\n * to the SafeSubscriber -- meaning no error handler was supplied\n * do the `subscribe` call on our observable.\n * @param err The error to handle\n */\nfunction defaultErrorHandler(err: any) {\n throw err;\n}\n\n/**\n * A handler for notifications that cannot be sent to a stopped subscriber.\n * @param notification The notification being sent\n * @param subscriber The stopped subscriber\n */\nfunction handleStoppedNotification(notification: ObservableNotification, subscriber: Subscriber) {\n const { onStoppedNotification } = config;\n onStoppedNotification && timeoutProvider.setTimeout(() => onStoppedNotification(notification, subscriber));\n}\n\n/**\n * The observer used as a stub for subscriptions where the user did not\n * pass any arguments to `subscribe`. Comes with the default error handling\n * behavior.\n */\nexport const EMPTY_OBSERVER: Readonly> & { closed: true } = {\n closed: true,\n next: noop,\n error: defaultErrorHandler,\n complete: noop,\n};\n", "/**\n * Symbol.observable or a string \"@@observable\". Used for interop\n *\n * @deprecated We will no longer be exporting this symbol in upcoming versions of RxJS.\n * Instead polyfill and use Symbol.observable directly *or* use https://www.npmjs.com/package/symbol-observable\n */\nexport const observable: string | symbol = (() => (typeof Symbol === 'function' && Symbol.observable) || '@@observable')();\n", "/**\n * This function takes one parameter and just returns it. Simply put,\n * this is like `(x: T): T => x`.\n *\n * ## Examples\n *\n * This is useful in some cases when using things like `mergeMap`\n *\n * ```ts\n * import { interval, take, map, range, mergeMap, identity } from 'rxjs';\n *\n * const source$ = interval(1000).pipe(take(5));\n *\n * const result$ = source$.pipe(\n * map(i => range(i)),\n * mergeMap(identity) // same as mergeMap(x => x)\n * );\n *\n * result$.subscribe({\n * next: console.log\n * });\n * ```\n *\n * Or when you want to selectively apply an operator\n *\n * ```ts\n * import { interval, take, identity } from 'rxjs';\n *\n * const shouldLimit = () => Math.random() < 0.5;\n *\n * const source$ = interval(1000);\n *\n * const result$ = source$.pipe(shouldLimit() ? take(5) : identity);\n *\n * result$.subscribe({\n * next: console.log\n * });\n * ```\n *\n * @param x Any value that is returned by this function\n * @returns The value passed as the first parameter to this function\n */\nexport function identity(x: T): T {\n return x;\n}\n", "import { identity } from './identity';\nimport { UnaryFunction } from '../types';\n\nexport function pipe(): typeof identity;\nexport function pipe(fn1: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction): UnaryFunction;\nexport function pipe(fn1: UnaryFunction, fn2: UnaryFunction, fn3: UnaryFunction): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction,\n fn9: UnaryFunction\n): UnaryFunction;\nexport function pipe(\n fn1: UnaryFunction,\n fn2: UnaryFunction,\n fn3: UnaryFunction,\n fn4: UnaryFunction,\n fn5: UnaryFunction,\n fn6: UnaryFunction,\n fn7: UnaryFunction,\n fn8: UnaryFunction,\n fn9: UnaryFunction,\n ...fns: UnaryFunction[]\n): UnaryFunction;\n\n/**\n * pipe() can be called on one or more functions, each of which can take one argument (\"UnaryFunction\")\n * and uses it to return a value.\n * It returns a function that takes one argument, passes it to the first UnaryFunction, and then\n * passes the result to the next one, passes that result to the next one, and so on. \n */\nexport function pipe(...fns: Array>): UnaryFunction {\n return pipeFromArray(fns);\n}\n\n/** @internal */\nexport function pipeFromArray(fns: Array>): UnaryFunction {\n if (fns.length === 0) {\n return identity as UnaryFunction;\n }\n\n if (fns.length === 1) {\n return fns[0];\n }\n\n return function piped(input: T): R {\n return fns.reduce((prev: any, fn: UnaryFunction) => fn(prev), input as any);\n };\n}\n", "import { Operator } from './Operator';\nimport { SafeSubscriber, Subscriber } from './Subscriber';\nimport { isSubscription, Subscription } from './Subscription';\nimport { TeardownLogic, OperatorFunction, Subscribable, Observer } from './types';\nimport { observable as Symbol_observable } from './symbol/observable';\nimport { pipeFromArray } from './util/pipe';\nimport { config } from './config';\nimport { isFunction } from './util/isFunction';\nimport { errorContext } from './util/errorContext';\n\n/**\n * A representation of any set of values over any amount of time. This is the most basic building block\n * of RxJS.\n *\n * @class Observable\n */\nexport class Observable implements Subscribable {\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n */\n source: Observable | undefined;\n\n /**\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n */\n operator: Operator | undefined;\n\n /**\n * @constructor\n * @param {Function} subscribe the function that is called when the Observable is\n * initially subscribed to. This function is given a Subscriber, to which new values\n * can be `next`ed, or an `error` method can be called to raise an error, or\n * `complete` can be called to notify of a successful completion.\n */\n constructor(subscribe?: (this: Observable, subscriber: Subscriber) => TeardownLogic) {\n if (subscribe) {\n this._subscribe = subscribe;\n }\n }\n\n // HACK: Since TypeScript inherits static properties too, we have to\n // fight against TypeScript here so Subject can have a different static create signature\n /**\n * Creates a new Observable by calling the Observable constructor\n * @owner Observable\n * @method create\n * @param {Function} subscribe? the subscriber function to be passed to the Observable constructor\n * @return {Observable} a new observable\n * @nocollapse\n * @deprecated Use `new Observable()` instead. Will be removed in v8.\n */\n static create: (...args: any[]) => any = (subscribe?: (subscriber: Subscriber) => TeardownLogic) => {\n return new Observable(subscribe);\n };\n\n /**\n * Creates a new Observable, with this Observable instance as the source, and the passed\n * operator defined as the new observable's operator.\n * @method lift\n * @param operator the operator defining the operation to take on the observable\n * @return a new observable with the Operator applied\n * @deprecated Internal implementation detail, do not use directly. Will be made internal in v8.\n * If you have implemented an operator using `lift`, it is recommended that you create an\n * operator by simply returning `new Observable()` directly. See \"Creating new operators from\n * scratch\" section here: https://rxjs.dev/guide/operators\n */\n lift(operator?: Operator): Observable {\n const observable = new Observable();\n observable.source = this;\n observable.operator = operator;\n return observable;\n }\n\n subscribe(observerOrNext?: Partial> | ((value: T) => void)): Subscription;\n /** @deprecated Instead of passing separate callback arguments, use an observer argument. Signatures taking separate callback arguments will be removed in v8. Details: https://rxjs.dev/deprecations/subscribe-arguments */\n subscribe(next?: ((value: T) => void) | null, error?: ((error: any) => void) | null, complete?: (() => void) | null): Subscription;\n /**\n * Invokes an execution of an Observable and registers Observer handlers for notifications it will emit.\n *\n * Use it when you have all these Observables, but still nothing is happening.\n *\n * `subscribe` is not a regular operator, but a method that calls Observable's internal `subscribe` function. It\n * might be for example a function that you passed to Observable's constructor, but most of the time it is\n * a library implementation, which defines what will be emitted by an Observable, and when it be will emitted. This means\n * that calling `subscribe` is actually the moment when Observable starts its work, not when it is created, as it is often\n * the thought.\n *\n * Apart from starting the execution of an Observable, this method allows you to listen for values\n * that an Observable emits, as well as for when it completes or errors. You can achieve this in two\n * of the following ways.\n *\n * The first way is creating an object that implements {@link Observer} interface. It should have methods\n * defined by that interface, but note that it should be just a regular JavaScript object, which you can create\n * yourself in any way you want (ES6 class, classic function constructor, object literal etc.). In particular, do\n * not attempt to use any RxJS implementation details to create Observers - you don't need them. Remember also\n * that your object does not have to implement all methods. If you find yourself creating a method that doesn't\n * do anything, you can simply omit it. Note however, if the `error` method is not provided and an error happens,\n * it will be thrown asynchronously. Errors thrown asynchronously cannot be caught using `try`/`catch`. Instead,\n * use the {@link onUnhandledError} configuration option or use a runtime handler (like `window.onerror` or\n * `process.on('error)`) to be notified of unhandled errors. Because of this, it's recommended that you provide\n * an `error` method to avoid missing thrown errors.\n *\n * The second way is to give up on Observer object altogether and simply provide callback functions in place of its methods.\n * This means you can provide three functions as arguments to `subscribe`, where the first function is equivalent\n * of a `next` method, the second of an `error` method and the third of a `complete` method. Just as in case of an Observer,\n * if you do not need to listen for something, you can omit a function by passing `undefined` or `null`,\n * since `subscribe` recognizes these functions by where they were placed in function call. When it comes\n * to the `error` function, as with an Observer, if not provided, errors emitted by an Observable will be thrown asynchronously.\n *\n * You can, however, subscribe with no parameters at all. This may be the case where you're not interested in terminal events\n * and you also handled emissions internally by using operators (e.g. using `tap`).\n *\n * Whichever style of calling `subscribe` you use, in both cases it returns a Subscription object.\n * This object allows you to call `unsubscribe` on it, which in turn will stop the work that an Observable does and will clean\n * up all resources that an Observable used. Note that cancelling a subscription will not call `complete` callback\n * provided to `subscribe` function, which is reserved for a regular completion signal that comes from an Observable.\n *\n * Remember that callbacks provided to `subscribe` are not guaranteed to be called asynchronously.\n * It is an Observable itself that decides when these functions will be called. For example {@link of}\n * by default emits all its values synchronously. Always check documentation for how given Observable\n * will behave when subscribed and if its default behavior can be modified with a `scheduler`.\n *\n * #### Examples\n *\n * Subscribe with an {@link guide/observer Observer}\n *\n * ```ts\n * import { of } from 'rxjs';\n *\n * const sumObserver = {\n * sum: 0,\n * next(value) {\n * console.log('Adding: ' + value);\n * this.sum = this.sum + value;\n * },\n * error() {\n * // We actually could just remove this method,\n * // since we do not really care about errors right now.\n * },\n * complete() {\n * console.log('Sum equals: ' + this.sum);\n * }\n * };\n *\n * of(1, 2, 3) // Synchronously emits 1, 2, 3 and then completes.\n * .subscribe(sumObserver);\n *\n * // Logs:\n * // 'Adding: 1'\n * // 'Adding: 2'\n * // 'Adding: 3'\n * // 'Sum equals: 6'\n * ```\n *\n * Subscribe with functions ({@link deprecations/subscribe-arguments deprecated})\n *\n * ```ts\n * import { of } from 'rxjs'\n *\n * let sum = 0;\n *\n * of(1, 2, 3).subscribe(\n * value => {\n * console.log('Adding: ' + value);\n * sum = sum + value;\n * },\n * undefined,\n * () => console.log('Sum equals: ' + sum)\n * );\n *\n * // Logs:\n * // 'Adding: 1'\n * // 'Adding: 2'\n * // 'Adding: 3'\n * // 'Sum equals: 6'\n * ```\n *\n * Cancel a subscription\n *\n * ```ts\n * import { interval } from 'rxjs';\n *\n * const subscription = interval(1000).subscribe({\n * next(num) {\n * console.log(num)\n * },\n * complete() {\n * // Will not be called, even when cancelling subscription.\n * console.log('completed!');\n * }\n * });\n *\n * setTimeout(() => {\n * subscription.unsubscribe();\n * console.log('unsubscribed!');\n * }, 2500);\n *\n * // Logs:\n * // 0 after 1s\n * // 1 after 2s\n * // 'unsubscribed!' after 2.5s\n * ```\n *\n * @param {Observer|Function} observerOrNext (optional) Either an observer with methods to be called,\n * or the first of three possible handlers, which is the handler for each value emitted from the subscribed\n * Observable.\n * @param {Function} error (optional) A handler for a terminal event resulting from an error. If no error handler is provided,\n * the error will be thrown asynchronously as unhandled.\n * @param {Function} complete (optional) A handler for a terminal event resulting from successful completion.\n * @return {Subscription} a subscription reference to the registered handlers\n * @method subscribe\n */\n subscribe(\n observerOrNext?: Partial> | ((value: T) => void) | null,\n error?: ((error: any) => void) | null,\n complete?: (() => void) | null\n ): Subscription {\n const subscriber = isSubscriber(observerOrNext) ? observerOrNext : new SafeSubscriber(observerOrNext, error, complete);\n\n errorContext(() => {\n const { operator, source } = this;\n subscriber.add(\n operator\n ? // We're dealing with a subscription in the\n // operator chain to one of our lifted operators.\n operator.call(subscriber, source)\n : source\n ? // If `source` has a value, but `operator` does not, something that\n // had intimate knowledge of our API, like our `Subject`, must have\n // set it. We're going to just call `_subscribe` directly.\n this._subscribe(subscriber)\n : // In all other cases, we're likely wrapping a user-provided initializer\n // function, so we need to catch errors and handle them appropriately.\n this._trySubscribe(subscriber)\n );\n });\n\n return subscriber;\n }\n\n /** @internal */\n protected _trySubscribe(sink: Subscriber): TeardownLogic {\n try {\n return this._subscribe(sink);\n } catch (err) {\n // We don't need to return anything in this case,\n // because it's just going to try to `add()` to a subscription\n // above.\n sink.error(err);\n }\n }\n\n /**\n * Used as a NON-CANCELLABLE means of subscribing to an observable, for use with\n * APIs that expect promises, like `async/await`. You cannot unsubscribe from this.\n *\n * **WARNING**: Only use this with observables you *know* will complete. If the source\n * observable does not complete, you will end up with a promise that is hung up, and\n * potentially all of the state of an async function hanging out in memory. To avoid\n * this situation, look into adding something like {@link timeout}, {@link take},\n * {@link takeWhile}, or {@link takeUntil} amongst others.\n *\n * #### Example\n *\n * ```ts\n * import { interval, take } from 'rxjs';\n *\n * const source$ = interval(1000).pipe(take(4));\n *\n * async function getTotal() {\n * let total = 0;\n *\n * await source$.forEach(value => {\n * total += value;\n * console.log('observable -> ' + value);\n * });\n *\n * return total;\n * }\n *\n * getTotal().then(\n * total => console.log('Total: ' + total)\n * );\n *\n * // Expected:\n * // 'observable -> 0'\n * // 'observable -> 1'\n * // 'observable -> 2'\n * // 'observable -> 3'\n * // 'Total: 6'\n * ```\n *\n * @param next a handler for each value emitted by the observable\n * @return a promise that either resolves on observable completion or\n * rejects with the handled error\n */\n forEach(next: (value: T) => void): Promise;\n\n /**\n * @param next a handler for each value emitted by the observable\n * @param promiseCtor a constructor function used to instantiate the Promise\n * @return a promise that either resolves on observable completion or\n * rejects with the handled error\n * @deprecated Passing a Promise constructor will no longer be available\n * in upcoming versions of RxJS. This is because it adds weight to the library, for very\n * little benefit. If you need this functionality, it is recommended that you either\n * polyfill Promise, or you create an adapter to convert the returned native promise\n * to whatever promise implementation you wanted. Will be removed in v8.\n */\n forEach(next: (value: T) => void, promiseCtor: PromiseConstructorLike): Promise;\n\n forEach(next: (value: T) => void, promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n const subscriber = new SafeSubscriber({\n next: (value) => {\n try {\n next(value);\n } catch (err) {\n reject(err);\n subscriber.unsubscribe();\n }\n },\n error: reject,\n complete: resolve,\n });\n this.subscribe(subscriber);\n }) as Promise;\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): TeardownLogic {\n return this.source?.subscribe(subscriber);\n }\n\n /**\n * An interop point defined by the es7-observable spec https://github.com/zenparsing/es-observable\n * @method Symbol.observable\n * @return {Observable} this instance of the observable\n */\n [Symbol_observable]() {\n return this;\n }\n\n /* tslint:disable:max-line-length */\n pipe(): Observable;\n pipe(op1: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction): Observable;\n pipe(op1: OperatorFunction, op2: OperatorFunction, op3: OperatorFunction): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction,\n op9: OperatorFunction\n ): Observable;\n pipe(\n op1: OperatorFunction,\n op2: OperatorFunction,\n op3: OperatorFunction,\n op4: OperatorFunction,\n op5: OperatorFunction,\n op6: OperatorFunction,\n op7: OperatorFunction,\n op8: OperatorFunction,\n op9: OperatorFunction,\n ...operations: OperatorFunction[]\n ): Observable;\n /* tslint:enable:max-line-length */\n\n /**\n * Used to stitch together functional operators into a chain.\n * @method pipe\n * @return {Observable} the Observable result of all of the operators having\n * been called in the order they were passed in.\n *\n * ## Example\n *\n * ```ts\n * import { interval, filter, map, scan } from 'rxjs';\n *\n * interval(1000)\n * .pipe(\n * filter(x => x % 2 === 0),\n * map(x => x + x),\n * scan((acc, x) => acc + x)\n * )\n * .subscribe(x => console.log(x));\n * ```\n */\n pipe(...operations: OperatorFunction[]): Observable {\n return pipeFromArray(operations)(this);\n }\n\n /* tslint:disable:max-line-length */\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(): Promise;\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(PromiseCtor: typeof Promise): Promise;\n /** @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise */\n toPromise(PromiseCtor: PromiseConstructorLike): Promise;\n /* tslint:enable:max-line-length */\n\n /**\n * Subscribe to this Observable and get a Promise resolving on\n * `complete` with the last emission (if any).\n *\n * **WARNING**: Only use this with observables you *know* will complete. If the source\n * observable does not complete, you will end up with a promise that is hung up, and\n * potentially all of the state of an async function hanging out in memory. To avoid\n * this situation, look into adding something like {@link timeout}, {@link take},\n * {@link takeWhile}, or {@link takeUntil} amongst others.\n *\n * @method toPromise\n * @param [promiseCtor] a constructor function used to instantiate\n * the Promise\n * @return A Promise that resolves with the last value emit, or\n * rejects on an error. If there were no emissions, Promise\n * resolves with undefined.\n * @deprecated Replaced with {@link firstValueFrom} and {@link lastValueFrom}. Will be removed in v8. Details: https://rxjs.dev/deprecations/to-promise\n */\n toPromise(promiseCtor?: PromiseConstructorLike): Promise {\n promiseCtor = getPromiseCtor(promiseCtor);\n\n return new promiseCtor((resolve, reject) => {\n let value: T | undefined;\n this.subscribe(\n (x: T) => (value = x),\n (err: any) => reject(err),\n () => resolve(value)\n );\n }) as Promise;\n }\n}\n\n/**\n * Decides between a passed promise constructor from consuming code,\n * A default configured promise constructor, and the native promise\n * constructor and returns it. If nothing can be found, it will throw\n * an error.\n * @param promiseCtor The optional promise constructor to passed by consuming code\n */\nfunction getPromiseCtor(promiseCtor: PromiseConstructorLike | undefined) {\n return promiseCtor ?? config.Promise ?? Promise;\n}\n\nfunction isObserver(value: any): value is Observer {\n return value && isFunction(value.next) && isFunction(value.error) && isFunction(value.complete);\n}\n\nfunction isSubscriber(value: any): value is Subscriber {\n return (value && value instanceof Subscriber) || (isObserver(value) && isSubscription(value));\n}\n", "import { Observable } from '../Observable';\nimport { Subscriber } from '../Subscriber';\nimport { OperatorFunction } from '../types';\nimport { isFunction } from './isFunction';\n\n/**\n * Used to determine if an object is an Observable with a lift function.\n */\nexport function hasLift(source: any): source is { lift: InstanceType['lift'] } {\n return isFunction(source?.lift);\n}\n\n/**\n * Creates an `OperatorFunction`. Used to define operators throughout the library in a concise way.\n * @param init The logic to connect the liftedSource to the subscriber at the moment of subscription.\n */\nexport function operate(\n init: (liftedSource: Observable, subscriber: Subscriber) => (() => void) | void\n): OperatorFunction {\n return (source: Observable) => {\n if (hasLift(source)) {\n return source.lift(function (this: Subscriber, liftedSource: Observable) {\n try {\n return init(liftedSource, this);\n } catch (err) {\n this.error(err);\n }\n });\n }\n throw new TypeError('Unable to lift unknown Observable type');\n };\n}\n", "import { Subscriber } from '../Subscriber';\n\n/**\n * Creates an instance of an `OperatorSubscriber`.\n * @param destination The downstream subscriber.\n * @param onNext Handles next values, only called if this subscriber is not stopped or closed. Any\n * error that occurs in this function is caught and sent to the `error` method of this subscriber.\n * @param onError Handles errors from the subscription, any errors that occur in this handler are caught\n * and send to the `destination` error handler.\n * @param onComplete Handles completion notification from the subscription. Any errors that occur in\n * this handler are sent to the `destination` error handler.\n * @param onFinalize Additional teardown logic here. This will only be called on teardown if the\n * subscriber itself is not already closed. This is called after all other teardown logic is executed.\n */\nexport function createOperatorSubscriber(\n destination: Subscriber,\n onNext?: (value: T) => void,\n onComplete?: () => void,\n onError?: (err: any) => void,\n onFinalize?: () => void\n): Subscriber {\n return new OperatorSubscriber(destination, onNext, onComplete, onError, onFinalize);\n}\n\n/**\n * A generic helper for allowing operators to be created with a Subscriber and\n * use closures to capture necessary state from the operator function itself.\n */\nexport class OperatorSubscriber extends Subscriber {\n /**\n * Creates an instance of an `OperatorSubscriber`.\n * @param destination The downstream subscriber.\n * @param onNext Handles next values, only called if this subscriber is not stopped or closed. Any\n * error that occurs in this function is caught and sent to the `error` method of this subscriber.\n * @param onError Handles errors from the subscription, any errors that occur in this handler are caught\n * and send to the `destination` error handler.\n * @param onComplete Handles completion notification from the subscription. Any errors that occur in\n * this handler are sent to the `destination` error handler.\n * @param onFinalize Additional finalization logic here. This will only be called on finalization if the\n * subscriber itself is not already closed. This is called after all other finalization logic is executed.\n * @param shouldUnsubscribe An optional check to see if an unsubscribe call should truly unsubscribe.\n * NOTE: This currently **ONLY** exists to support the strange behavior of {@link groupBy}, where unsubscription\n * to the resulting observable does not actually disconnect from the source if there are active subscriptions\n * to any grouped observable. (DO NOT EXPOSE OR USE EXTERNALLY!!!)\n */\n constructor(\n destination: Subscriber,\n onNext?: (value: T) => void,\n onComplete?: () => void,\n onError?: (err: any) => void,\n private onFinalize?: () => void,\n private shouldUnsubscribe?: () => boolean\n ) {\n // It's important - for performance reasons - that all of this class's\n // members are initialized and that they are always initialized in the same\n // order. This will ensure that all OperatorSubscriber instances have the\n // same hidden class in V8. This, in turn, will help keep the number of\n // hidden classes involved in property accesses within the base class as\n // low as possible. If the number of hidden classes involved exceeds four,\n // the property accesses will become megamorphic and performance penalties\n // will be incurred - i.e. inline caches won't be used.\n //\n // The reasons for ensuring all instances have the same hidden class are\n // further discussed in this blog post from Benedikt Meurer:\n // https://benediktmeurer.de/2018/03/23/impact-of-polymorphism-on-component-based-frameworks-like-react/\n super(destination);\n this._next = onNext\n ? function (this: OperatorSubscriber, value: T) {\n try {\n onNext(value);\n } catch (err) {\n destination.error(err);\n }\n }\n : super._next;\n this._error = onError\n ? function (this: OperatorSubscriber, err: any) {\n try {\n onError(err);\n } catch (err) {\n // Send any errors that occur down stream.\n destination.error(err);\n } finally {\n // Ensure finalization.\n this.unsubscribe();\n }\n }\n : super._error;\n this._complete = onComplete\n ? function (this: OperatorSubscriber) {\n try {\n onComplete();\n } catch (err) {\n // Send any errors that occur down stream.\n destination.error(err);\n } finally {\n // Ensure finalization.\n this.unsubscribe();\n }\n }\n : super._complete;\n }\n\n unsubscribe() {\n if (!this.shouldUnsubscribe || this.shouldUnsubscribe()) {\n const { closed } = this;\n super.unsubscribe();\n // Execute additional teardown if we have any and we didn't already do so.\n !closed && this.onFinalize?.();\n }\n }\n}\n", "import { Subscription } from '../Subscription';\n\ninterface AnimationFrameProvider {\n schedule(callback: FrameRequestCallback): Subscription;\n requestAnimationFrame: typeof requestAnimationFrame;\n cancelAnimationFrame: typeof cancelAnimationFrame;\n delegate:\n | {\n requestAnimationFrame: typeof requestAnimationFrame;\n cancelAnimationFrame: typeof cancelAnimationFrame;\n }\n | undefined;\n}\n\nexport const animationFrameProvider: AnimationFrameProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n schedule(callback) {\n let request = requestAnimationFrame;\n let cancel: typeof cancelAnimationFrame | undefined = cancelAnimationFrame;\n const { delegate } = animationFrameProvider;\n if (delegate) {\n request = delegate.requestAnimationFrame;\n cancel = delegate.cancelAnimationFrame;\n }\n const handle = request((timestamp) => {\n // Clear the cancel function. The request has been fulfilled, so\n // attempting to cancel the request upon unsubscription would be\n // pointless.\n cancel = undefined;\n callback(timestamp);\n });\n return new Subscription(() => cancel?.(handle));\n },\n requestAnimationFrame(...args) {\n const { delegate } = animationFrameProvider;\n return (delegate?.requestAnimationFrame || requestAnimationFrame)(...args);\n },\n cancelAnimationFrame(...args) {\n const { delegate } = animationFrameProvider;\n return (delegate?.cancelAnimationFrame || cancelAnimationFrame)(...args);\n },\n delegate: undefined,\n};\n", "import { createErrorClass } from './createErrorClass';\n\nexport interface ObjectUnsubscribedError extends Error {}\n\nexport interface ObjectUnsubscribedErrorCtor {\n /**\n * @deprecated Internal implementation detail. Do not construct error instances.\n * Cannot be tagged as internal: https://github.com/ReactiveX/rxjs/issues/6269\n */\n new (): ObjectUnsubscribedError;\n}\n\n/**\n * An error thrown when an action is invalid because the object has been\n * unsubscribed.\n *\n * @see {@link Subject}\n * @see {@link BehaviorSubject}\n *\n * @class ObjectUnsubscribedError\n */\nexport const ObjectUnsubscribedError: ObjectUnsubscribedErrorCtor = createErrorClass(\n (_super) =>\n function ObjectUnsubscribedErrorImpl(this: any) {\n _super(this);\n this.name = 'ObjectUnsubscribedError';\n this.message = 'object unsubscribed';\n }\n);\n", "import { Operator } from './Operator';\nimport { Observable } from './Observable';\nimport { Subscriber } from './Subscriber';\nimport { Subscription, EMPTY_SUBSCRIPTION } from './Subscription';\nimport { Observer, SubscriptionLike, TeardownLogic } from './types';\nimport { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';\nimport { arrRemove } from './util/arrRemove';\nimport { errorContext } from './util/errorContext';\n\n/**\n * A Subject is a special type of Observable that allows values to be\n * multicasted to many Observers. Subjects are like EventEmitters.\n *\n * Every Subject is an Observable and an Observer. You can subscribe to a\n * Subject, and you can call next to feed values as well as error and complete.\n */\nexport class Subject extends Observable implements SubscriptionLike {\n closed = false;\n\n private currentObservers: Observer[] | null = null;\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n observers: Observer[] = [];\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n isStopped = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n hasError = false;\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n thrownError: any = null;\n\n /**\n * Creates a \"subject\" by basically gluing an observer to an observable.\n *\n * @nocollapse\n * @deprecated Recommended you do not use. Will be removed at some point in the future. Plans for replacement still under discussion.\n */\n static create: (...args: any[]) => any = (destination: Observer, source: Observable): AnonymousSubject => {\n return new AnonymousSubject(destination, source);\n };\n\n constructor() {\n // NOTE: This must be here to obscure Observable's constructor.\n super();\n }\n\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n lift(operator: Operator): Observable {\n const subject = new AnonymousSubject(this, this);\n subject.operator = operator as any;\n return subject as any;\n }\n\n /** @internal */\n protected _throwIfClosed() {\n if (this.closed) {\n throw new ObjectUnsubscribedError();\n }\n }\n\n next(value: T) {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n if (!this.currentObservers) {\n this.currentObservers = Array.from(this.observers);\n }\n for (const observer of this.currentObservers) {\n observer.next(value);\n }\n }\n });\n }\n\n error(err: any) {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n this.hasError = this.isStopped = true;\n this.thrownError = err;\n const { observers } = this;\n while (observers.length) {\n observers.shift()!.error(err);\n }\n }\n });\n }\n\n complete() {\n errorContext(() => {\n this._throwIfClosed();\n if (!this.isStopped) {\n this.isStopped = true;\n const { observers } = this;\n while (observers.length) {\n observers.shift()!.complete();\n }\n }\n });\n }\n\n unsubscribe() {\n this.isStopped = this.closed = true;\n this.observers = this.currentObservers = null!;\n }\n\n get observed() {\n return this.observers?.length > 0;\n }\n\n /** @internal */\n protected _trySubscribe(subscriber: Subscriber): TeardownLogic {\n this._throwIfClosed();\n return super._trySubscribe(subscriber);\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n this._throwIfClosed();\n this._checkFinalizedStatuses(subscriber);\n return this._innerSubscribe(subscriber);\n }\n\n /** @internal */\n protected _innerSubscribe(subscriber: Subscriber) {\n const { hasError, isStopped, observers } = this;\n if (hasError || isStopped) {\n return EMPTY_SUBSCRIPTION;\n }\n this.currentObservers = null;\n observers.push(subscriber);\n return new Subscription(() => {\n this.currentObservers = null;\n arrRemove(observers, subscriber);\n });\n }\n\n /** @internal */\n protected _checkFinalizedStatuses(subscriber: Subscriber) {\n const { hasError, thrownError, isStopped } = this;\n if (hasError) {\n subscriber.error(thrownError);\n } else if (isStopped) {\n subscriber.complete();\n }\n }\n\n /**\n * Creates a new Observable with this Subject as the source. You can do this\n * to create custom Observer-side logic of the Subject and conceal it from\n * code that uses the Observable.\n * @return {Observable} Observable that the Subject casts to\n */\n asObservable(): Observable {\n const observable: any = new Observable();\n observable.source = this;\n return observable;\n }\n}\n\n/**\n * @class AnonymousSubject\n */\nexport class AnonymousSubject extends Subject {\n constructor(\n /** @deprecated Internal implementation detail, do not use directly. Will be made internal in v8. */\n public destination?: Observer,\n source?: Observable\n ) {\n super();\n this.source = source;\n }\n\n next(value: T) {\n this.destination?.next?.(value);\n }\n\n error(err: any) {\n this.destination?.error?.(err);\n }\n\n complete() {\n this.destination?.complete?.();\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n return this.source?.subscribe(subscriber) ?? EMPTY_SUBSCRIPTION;\n }\n}\n", "import { Subject } from './Subject';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\n\n/**\n * A variant of Subject that requires an initial value and emits its current\n * value whenever it is subscribed to.\n *\n * @class BehaviorSubject\n */\nexport class BehaviorSubject extends Subject {\n constructor(private _value: T) {\n super();\n }\n\n get value(): T {\n return this.getValue();\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n const subscription = super._subscribe(subscriber);\n !subscription.closed && subscriber.next(this._value);\n return subscription;\n }\n\n getValue(): T {\n const { hasError, thrownError, _value } = this;\n if (hasError) {\n throw thrownError;\n }\n this._throwIfClosed();\n return _value;\n }\n\n next(value: T): void {\n super.next((this._value = value));\n }\n}\n", "import { TimestampProvider } from '../types';\n\ninterface DateTimestampProvider extends TimestampProvider {\n delegate: TimestampProvider | undefined;\n}\n\nexport const dateTimestampProvider: DateTimestampProvider = {\n now() {\n // Use the variable rather than `this` so that the function can be called\n // without being bound to the provider.\n return (dateTimestampProvider.delegate || Date).now();\n },\n delegate: undefined,\n};\n", "import { Subject } from './Subject';\nimport { TimestampProvider } from './types';\nimport { Subscriber } from './Subscriber';\nimport { Subscription } from './Subscription';\nimport { dateTimestampProvider } from './scheduler/dateTimestampProvider';\n\n/**\n * A variant of {@link Subject} that \"replays\" old values to new subscribers by emitting them when they first subscribe.\n *\n * `ReplaySubject` has an internal buffer that will store a specified number of values that it has observed. Like `Subject`,\n * `ReplaySubject` \"observes\" values by having them passed to its `next` method. When it observes a value, it will store that\n * value for a time determined by the configuration of the `ReplaySubject`, as passed to its constructor.\n *\n * When a new subscriber subscribes to the `ReplaySubject` instance, it will synchronously emit all values in its buffer in\n * a First-In-First-Out (FIFO) manner. The `ReplaySubject` will also complete, if it has observed completion; and it will\n * error if it has observed an error.\n *\n * There are two main configuration items to be concerned with:\n *\n * 1. `bufferSize` - This will determine how many items are stored in the buffer, defaults to infinite.\n * 2. `windowTime` - The amount of time to hold a value in the buffer before removing it from the buffer.\n *\n * Both configurations may exist simultaneously. So if you would like to buffer a maximum of 3 values, as long as the values\n * are less than 2 seconds old, you could do so with a `new ReplaySubject(3, 2000)`.\n *\n * ### Differences with BehaviorSubject\n *\n * `BehaviorSubject` is similar to `new ReplaySubject(1)`, with a couple of exceptions:\n *\n * 1. `BehaviorSubject` comes \"primed\" with a single value upon construction.\n * 2. `ReplaySubject` will replay values, even after observing an error, where `BehaviorSubject` will not.\n *\n * @see {@link Subject}\n * @see {@link BehaviorSubject}\n * @see {@link shareReplay}\n */\nexport class ReplaySubject extends Subject {\n private _buffer: (T | number)[] = [];\n private _infiniteTimeWindow = true;\n\n /**\n * @param bufferSize The size of the buffer to replay on subscription\n * @param windowTime The amount of time the buffered items will stay buffered\n * @param timestampProvider An object with a `now()` method that provides the current timestamp. This is used to\n * calculate the amount of time something has been buffered.\n */\n constructor(\n private _bufferSize = Infinity,\n private _windowTime = Infinity,\n private _timestampProvider: TimestampProvider = dateTimestampProvider\n ) {\n super();\n this._infiniteTimeWindow = _windowTime === Infinity;\n this._bufferSize = Math.max(1, _bufferSize);\n this._windowTime = Math.max(1, _windowTime);\n }\n\n next(value: T): void {\n const { isStopped, _buffer, _infiniteTimeWindow, _timestampProvider, _windowTime } = this;\n if (!isStopped) {\n _buffer.push(value);\n !_infiniteTimeWindow && _buffer.push(_timestampProvider.now() + _windowTime);\n }\n this._trimBuffer();\n super.next(value);\n }\n\n /** @internal */\n protected _subscribe(subscriber: Subscriber): Subscription {\n this._throwIfClosed();\n this._trimBuffer();\n\n const subscription = this._innerSubscribe(subscriber);\n\n const { _infiniteTimeWindow, _buffer } = this;\n // We use a copy here, so reentrant code does not mutate our array while we're\n // emitting it to a new subscriber.\n const copy = _buffer.slice();\n for (let i = 0; i < copy.length && !subscriber.closed; i += _infiniteTimeWindow ? 1 : 2) {\n subscriber.next(copy[i] as T);\n }\n\n this._checkFinalizedStatuses(subscriber);\n\n return subscription;\n }\n\n private _trimBuffer() {\n const { _bufferSize, _timestampProvider, _buffer, _infiniteTimeWindow } = this;\n // If we don't have an infinite buffer size, and we're over the length,\n // use splice to truncate the old buffer values off. Note that we have to\n // double the size for instances where we're not using an infinite time window\n // because we're storing the values and the timestamps in the same array.\n const adjustedBufferSize = (_infiniteTimeWindow ? 1 : 2) * _bufferSize;\n _bufferSize < Infinity && adjustedBufferSize < _buffer.length && _buffer.splice(0, _buffer.length - adjustedBufferSize);\n\n // Now, if we're not in an infinite time window, remove all values where the time is\n // older than what is allowed.\n if (!_infiniteTimeWindow) {\n const now = _timestampProvider.now();\n let last = 0;\n // Search the array for the first timestamp that isn't expired and\n // truncate the buffer up to that point.\n for (let i = 1; i < _buffer.length && (_buffer[i] as number) <= now; i += 2) {\n last = i;\n }\n last && _buffer.splice(0, last + 1);\n }\n }\n}\n", "import { Scheduler } from '../Scheduler';\nimport { Subscription } from '../Subscription';\nimport { SchedulerAction } from '../types';\n\n/**\n * A unit of work to be executed in a `scheduler`. An action is typically\n * created from within a {@link SchedulerLike} and an RxJS user does not need to concern\n * themselves about creating and manipulating an Action.\n *\n * ```ts\n * class Action extends Subscription {\n * new (scheduler: Scheduler, work: (state?: T) => void);\n * schedule(state?: T, delay: number = 0): Subscription;\n * }\n * ```\n *\n * @class Action\n */\nexport class Action extends Subscription {\n constructor(scheduler: Scheduler, work: (this: SchedulerAction, state?: T) => void) {\n super();\n }\n /**\n * Schedules this action on its parent {@link SchedulerLike} for execution. May be passed\n * some context object, `state`. May happen at some point in the future,\n * according to the `delay` parameter, if specified.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler.\n * @return {void}\n */\n public schedule(state?: T, delay: number = 0): Subscription {\n return this;\n }\n}\n", "import type { TimerHandle } from './timerHandle';\ntype SetIntervalFunction = (handler: () => void, timeout?: number, ...args: any[]) => TimerHandle;\ntype ClearIntervalFunction = (handle: TimerHandle) => void;\n\ninterface IntervalProvider {\n setInterval: SetIntervalFunction;\n clearInterval: ClearIntervalFunction;\n delegate:\n | {\n setInterval: SetIntervalFunction;\n clearInterval: ClearIntervalFunction;\n }\n | undefined;\n}\n\nexport const intervalProvider: IntervalProvider = {\n // When accessing the delegate, use the variable rather than `this` so that\n // the functions can be called without being bound to the provider.\n setInterval(handler: () => void, timeout?: number, ...args) {\n const { delegate } = intervalProvider;\n if (delegate?.setInterval) {\n return delegate.setInterval(handler, timeout, ...args);\n }\n return setInterval(handler, timeout, ...args);\n },\n clearInterval(handle) {\n const { delegate } = intervalProvider;\n return (delegate?.clearInterval || clearInterval)(handle as any);\n },\n delegate: undefined,\n};\n", "import { Action } from './Action';\nimport { SchedulerAction } from '../types';\nimport { Subscription } from '../Subscription';\nimport { AsyncScheduler } from './AsyncScheduler';\nimport { intervalProvider } from './intervalProvider';\nimport { arrRemove } from '../util/arrRemove';\nimport { TimerHandle } from './timerHandle';\n\nexport class AsyncAction extends Action {\n public id: TimerHandle | undefined;\n public state?: T;\n // @ts-ignore: Property has no initializer and is not definitely assigned\n public delay: number;\n protected pending: boolean = false;\n\n constructor(protected scheduler: AsyncScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n if (this.closed) {\n return this;\n }\n\n // Always replace the current state with the new state.\n this.state = state;\n\n const id = this.id;\n const scheduler = this.scheduler;\n\n //\n // Important implementation note:\n //\n // Actions only execute once by default, unless rescheduled from within the\n // scheduled callback. This allows us to implement single and repeat\n // actions via the same code path, without adding API surface area, as well\n // as mimic traditional recursion but across asynchronous boundaries.\n //\n // However, JS runtimes and timers distinguish between intervals achieved by\n // serial `setTimeout` calls vs. a single `setInterval` call. An interval of\n // serial `setTimeout` calls can be individually delayed, which delays\n // scheduling the next `setTimeout`, and so on. `setInterval` attempts to\n // guarantee the interval callback will be invoked more precisely to the\n // interval period, regardless of load.\n //\n // Therefore, we use `setInterval` to schedule single and repeat actions.\n // If the action reschedules itself with the same delay, the interval is not\n // canceled. If the action doesn't reschedule, or reschedules with a\n // different delay, the interval will be canceled after scheduled callback\n // execution.\n //\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, delay);\n }\n\n // Set the pending flag indicating that this action has been scheduled, or\n // has recursively rescheduled itself.\n this.pending = true;\n\n this.delay = delay;\n // If this action has already an async Id, don't request a new one.\n this.id = this.id ?? this.requestAsyncId(scheduler, this.id, delay);\n\n return this;\n }\n\n protected requestAsyncId(scheduler: AsyncScheduler, _id?: TimerHandle, delay: number = 0): TimerHandle {\n return intervalProvider.setInterval(scheduler.flush.bind(scheduler, this), delay);\n }\n\n protected recycleAsyncId(_scheduler: AsyncScheduler, id?: TimerHandle, delay: number | null = 0): TimerHandle | undefined {\n // If this action is rescheduled with the same delay time, don't clear the interval id.\n if (delay != null && this.delay === delay && this.pending === false) {\n return id;\n }\n // Otherwise, if the action's delay time is different from the current delay,\n // or the action has been rescheduled before it's executed, clear the interval id\n if (id != null) {\n intervalProvider.clearInterval(id);\n }\n\n return undefined;\n }\n\n /**\n * Immediately executes this action and the `work` it contains.\n * @return {any}\n */\n public execute(state: T, delay: number): any {\n if (this.closed) {\n return new Error('executing a cancelled action');\n }\n\n this.pending = false;\n const error = this._execute(state, delay);\n if (error) {\n return error;\n } else if (this.pending === false && this.id != null) {\n // Dequeue if the action didn't reschedule itself. Don't call\n // unsubscribe(), because the action could reschedule later.\n // For example:\n // ```\n // scheduler.schedule(function doWork(counter) {\n // /* ... I'm a busy worker bee ... */\n // var originalAction = this;\n // /* wait 100ms before rescheduling the action */\n // setTimeout(function () {\n // originalAction.schedule(counter + 1);\n // }, 100);\n // }, 1000);\n // ```\n this.id = this.recycleAsyncId(this.scheduler, this.id, null);\n }\n }\n\n protected _execute(state: T, _delay: number): any {\n let errored: boolean = false;\n let errorValue: any;\n try {\n this.work(state);\n } catch (e) {\n errored = true;\n // HACK: Since code elsewhere is relying on the \"truthiness\" of the\n // return here, we can't have it return \"\" or 0 or false.\n // TODO: Clean this up when we refactor schedulers mid-version-8 or so.\n errorValue = e ? e : new Error('Scheduled action threw falsy error');\n }\n if (errored) {\n this.unsubscribe();\n return errorValue;\n }\n }\n\n unsubscribe() {\n if (!this.closed) {\n const { id, scheduler } = this;\n const { actions } = scheduler;\n\n this.work = this.state = this.scheduler = null!;\n this.pending = false;\n\n arrRemove(actions, this);\n if (id != null) {\n this.id = this.recycleAsyncId(scheduler, id, null);\n }\n\n this.delay = null!;\n super.unsubscribe();\n }\n }\n}\n", "import { Action } from './scheduler/Action';\nimport { Subscription } from './Subscription';\nimport { SchedulerLike, SchedulerAction } from './types';\nimport { dateTimestampProvider } from './scheduler/dateTimestampProvider';\n\n/**\n * An execution context and a data structure to order tasks and schedule their\n * execution. Provides a notion of (potentially virtual) time, through the\n * `now()` getter method.\n *\n * Each unit of work in a Scheduler is called an `Action`.\n *\n * ```ts\n * class Scheduler {\n * now(): number;\n * schedule(work, delay?, state?): Subscription;\n * }\n * ```\n *\n * @class Scheduler\n * @deprecated Scheduler is an internal implementation detail of RxJS, and\n * should not be used directly. Rather, create your own class and implement\n * {@link SchedulerLike}. Will be made internal in v8.\n */\nexport class Scheduler implements SchedulerLike {\n public static now: () => number = dateTimestampProvider.now;\n\n constructor(private schedulerActionCtor: typeof Action, now: () => number = Scheduler.now) {\n this.now = now;\n }\n\n /**\n * A getter method that returns a number representing the current time\n * (at the time this function was called) according to the scheduler's own\n * internal clock.\n * @return {number} A number that represents the current time. May or may not\n * have a relation to wall-clock time. May or may not refer to a time unit\n * (e.g. milliseconds).\n */\n public now: () => number;\n\n /**\n * Schedules a function, `work`, for execution. May happen at some point in\n * the future, according to the `delay` parameter, if specified. May be passed\n * some context object, `state`, which will be passed to the `work` function.\n *\n * The given arguments will be processed an stored as an Action object in a\n * queue of actions.\n *\n * @param {function(state: ?T): ?Subscription} work A function representing a\n * task, or some unit of work to be executed by the Scheduler.\n * @param {number} [delay] Time to wait before executing the work, where the\n * time unit is implicit and defined by the Scheduler itself.\n * @param {T} [state] Some contextual data that the `work` function uses when\n * called by the Scheduler.\n * @return {Subscription} A subscription in order to be able to unsubscribe\n * the scheduled work.\n */\n public schedule(work: (this: SchedulerAction, state?: T) => void, delay: number = 0, state?: T): Subscription {\n return new this.schedulerActionCtor(this, work).schedule(state, delay);\n }\n}\n", "import { Scheduler } from '../Scheduler';\nimport { Action } from './Action';\nimport { AsyncAction } from './AsyncAction';\nimport { TimerHandle } from './timerHandle';\n\nexport class AsyncScheduler extends Scheduler {\n public actions: Array> = [];\n /**\n * A flag to indicate whether the Scheduler is currently executing a batch of\n * queued actions.\n * @type {boolean}\n * @internal\n */\n public _active: boolean = false;\n /**\n * An internal ID used to track the latest asynchronous task such as those\n * coming from `setTimeout`, `setInterval`, `requestAnimationFrame`, and\n * others.\n * @type {any}\n * @internal\n */\n public _scheduled: TimerHandle | undefined;\n\n constructor(SchedulerAction: typeof Action, now: () => number = Scheduler.now) {\n super(SchedulerAction, now);\n }\n\n public flush(action: AsyncAction): void {\n const { actions } = this;\n\n if (this._active) {\n actions.push(action);\n return;\n }\n\n let error: any;\n this._active = true;\n\n do {\n if ((error = action.execute(action.state, action.delay))) {\n break;\n }\n } while ((action = actions.shift()!)); // exhaust the scheduler queue\n\n this._active = false;\n\n if (error) {\n while ((action = actions.shift()!)) {\n action.unsubscribe();\n }\n throw error;\n }\n }\n}\n", "import { AsyncAction } from './AsyncAction';\nimport { AsyncScheduler } from './AsyncScheduler';\n\n/**\n *\n * Async Scheduler\n *\n * Schedule task as if you used setTimeout(task, duration)\n *\n * `async` scheduler schedules tasks asynchronously, by putting them on the JavaScript\n * event loop queue. It is best used to delay tasks in time or to schedule tasks repeating\n * in intervals.\n *\n * If you just want to \"defer\" task, that is to perform it right after currently\n * executing synchronous code ends (commonly achieved by `setTimeout(deferredTask, 0)`),\n * better choice will be the {@link asapScheduler} scheduler.\n *\n * ## Examples\n * Use async scheduler to delay task\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * const task = () => console.log('it works!');\n *\n * asyncScheduler.schedule(task, 2000);\n *\n * // After 2 seconds logs:\n * // \"it works!\"\n * ```\n *\n * Use async scheduler to repeat task in intervals\n * ```ts\n * import { asyncScheduler } from 'rxjs';\n *\n * function task(state) {\n * console.log(state);\n * this.schedule(state + 1, 1000); // `this` references currently executing Action,\n * // which we reschedule with new state and delay\n * }\n *\n * asyncScheduler.schedule(task, 3000, 0);\n *\n * // Logs:\n * // 0 after 3s\n * // 1 after 4s\n * // 2 after 5s\n * // 3 after 6s\n * ```\n */\n\nexport const asyncScheduler = new AsyncScheduler(AsyncAction);\n\n/**\n * @deprecated Renamed to {@link asyncScheduler}. Will be removed in v8.\n */\nexport const async = asyncScheduler;\n", "import { AsyncAction } from './AsyncAction';\nimport { Subscription } from '../Subscription';\nimport { QueueScheduler } from './QueueScheduler';\nimport { SchedulerAction } from '../types';\nimport { TimerHandle } from './timerHandle';\n\nexport class QueueAction extends AsyncAction {\n constructor(protected scheduler: QueueScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n public schedule(state?: T, delay: number = 0): Subscription {\n if (delay > 0) {\n return super.schedule(state, delay);\n }\n this.delay = delay;\n this.state = state;\n this.scheduler.flush(this);\n return this;\n }\n\n public execute(state: T, delay: number): any {\n return delay > 0 || this.closed ? super.execute(state, delay) : this._execute(state, delay);\n }\n\n protected requestAsyncId(scheduler: QueueScheduler, id?: TimerHandle, delay: number = 0): TimerHandle {\n // If delay exists and is greater than 0, or if the delay is null (the\n // action wasn't rescheduled) but was originally scheduled as an async\n // action, then recycle as an async action.\n\n if ((delay != null && delay > 0) || (delay == null && this.delay > 0)) {\n return super.requestAsyncId(scheduler, id, delay);\n }\n\n // Otherwise flush the scheduler starting with this action.\n scheduler.flush(this);\n\n // HACK: In the past, this was returning `void`. However, `void` isn't a valid\n // `TimerHandle`, and generally the return value here isn't really used. So the\n // compromise is to return `0` which is both \"falsy\" and a valid `TimerHandle`,\n // as opposed to refactoring every other instanceo of `requestAsyncId`.\n return 0;\n }\n}\n", "import { AsyncScheduler } from './AsyncScheduler';\n\nexport class QueueScheduler extends AsyncScheduler {\n}\n", "import { QueueAction } from './QueueAction';\nimport { QueueScheduler } from './QueueScheduler';\n\n/**\n *\n * Queue Scheduler\n *\n * Put every next task on a queue, instead of executing it immediately\n *\n * `queue` scheduler, when used with delay, behaves the same as {@link asyncScheduler} scheduler.\n *\n * When used without delay, it schedules given task synchronously - executes it right when\n * it is scheduled. However when called recursively, that is when inside the scheduled task,\n * another task is scheduled with queue scheduler, instead of executing immediately as well,\n * that task will be put on a queue and wait for current one to finish.\n *\n * This means that when you execute task with `queue` scheduler, you are sure it will end\n * before any other task scheduled with that scheduler will start.\n *\n * ## Examples\n * Schedule recursively first, then do something\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(() => {\n * queueScheduler.schedule(() => console.log('second')); // will not happen now, but will be put on a queue\n *\n * console.log('first');\n * });\n *\n * // Logs:\n * // \"first\"\n * // \"second\"\n * ```\n *\n * Reschedule itself recursively\n * ```ts\n * import { queueScheduler } from 'rxjs';\n *\n * queueScheduler.schedule(function(state) {\n * if (state !== 0) {\n * console.log('before', state);\n * this.schedule(state - 1); // `this` references currently executing Action,\n * // which we reschedule with new state\n * console.log('after', state);\n * }\n * }, 0, 3);\n *\n * // In scheduler that runs recursively, you would expect:\n * // \"before\", 3\n * // \"before\", 2\n * // \"before\", 1\n * // \"after\", 1\n * // \"after\", 2\n * // \"after\", 3\n *\n * // But with queue it logs:\n * // \"before\", 3\n * // \"after\", 3\n * // \"before\", 2\n * // \"after\", 2\n * // \"before\", 1\n * // \"after\", 1\n * ```\n */\n\nexport const queueScheduler = new QueueScheduler(QueueAction);\n\n/**\n * @deprecated Renamed to {@link queueScheduler}. Will be removed in v8.\n */\nexport const queue = queueScheduler;\n", "import { AsyncAction } from './AsyncAction';\nimport { AnimationFrameScheduler } from './AnimationFrameScheduler';\nimport { SchedulerAction } from '../types';\nimport { animationFrameProvider } from './animationFrameProvider';\nimport { TimerHandle } from './timerHandle';\n\nexport class AnimationFrameAction extends AsyncAction {\n constructor(protected scheduler: AnimationFrameScheduler, protected work: (this: SchedulerAction, state?: T) => void) {\n super(scheduler, work);\n }\n\n protected requestAsyncId(scheduler: AnimationFrameScheduler, id?: TimerHandle, delay: number = 0): TimerHandle {\n // If delay is greater than 0, request as an async action.\n if (delay !== null && delay > 0) {\n return super.requestAsyncId(scheduler, id, delay);\n }\n // Push the action to the end of the scheduler queue.\n scheduler.actions.push(this);\n // If an animation frame has already been requested, don't request another\n // one. If an animation frame hasn't been requested yet, request one. Return\n // the current animation frame request id.\n return scheduler._scheduled || (scheduler._scheduled = animationFrameProvider.requestAnimationFrame(() => scheduler.flush(undefined)));\n }\n\n protected recycleAsyncId(scheduler: AnimationFrameScheduler, id?: TimerHandle, delay: number = 0): TimerHandle | undefined {\n // If delay exists and is greater than 0, or if the delay is null (the\n // action wasn't rescheduled) but was originally scheduled as an async\n // action, then recycle as an async action.\n if (delay != null ? delay > 0 : this.delay > 0) {\n return super.recycleAsyncId(scheduler, id, delay);\n }\n // If the scheduler queue has no remaining actions with the same async id,\n // cancel the requested animation frame and set the scheduled flag to\n // undefined so the next AnimationFrameAction will request its own.\n const { actions } = scheduler;\n if (id != null && actions[actions.length - 1]?.id !== id) {\n animationFrameProvider.cancelAnimationFrame(id as number);\n scheduler._scheduled = undefined;\n }\n // Return undefined so the action knows to request a new async id if it's rescheduled.\n return undefined;\n }\n}\n", "import { AsyncAction } from './AsyncAction';\nimport { AsyncScheduler } from './AsyncScheduler';\n\nexport class AnimationFrameScheduler extends AsyncScheduler {\n public flush(action?: AsyncAction): void {\n this._active = true;\n // The async id that effects a call to flush is stored in _scheduled.\n // Before executing an action, it's necessary to check the action's async\n // id to determine whether it's supposed to be executed in the current\n // flush.\n // Previous implementations of this method used a count to determine this,\n // but that was unsound, as actions that are unsubscribed - i.e. cancelled -\n // are removed from the actions array and that can shift actions that are\n // scheduled to be executed in a subsequent flush into positions at which\n // they are executed within the current flush.\n const flushId = this._scheduled;\n this._scheduled = undefined;\n\n const { actions } = this;\n let error: any;\n action = action || actions.shift()!;\n\n do {\n if ((error = action.execute(action.state, action.delay))) {\n break;\n }\n } while ((action = actions[0]) && action.id === flushId && actions.shift());\n\n this._active = false;\n\n if (error) {\n while ((action = actions[0]) && action.id === flushId && actions.shift()) {\n action.unsubscribe();\n }\n throw error;\n }\n }\n}\n", "import { AnimationFrameAction } from './AnimationFrameAction';\nimport { AnimationFrameScheduler } from './AnimationFrameScheduler';\n\n/**\n *\n * Animation Frame Scheduler\n *\n * Perform task when `window.requestAnimationFrame` would fire\n *\n * When `animationFrame` scheduler is used with delay, it will fall back to {@link asyncScheduler} scheduler\n * behaviour.\n *\n * Without delay, `animationFrame` scheduler can be used to create smooth browser animations.\n * It makes sure scheduled task will happen just before next browser content repaint,\n * thus performing animations as efficiently as possible.\n *\n * ## Example\n * Schedule div height animation\n * ```ts\n * // html:
\n * import { animationFrameScheduler } from 'rxjs';\n *\n * const div = document.querySelector('div');\n *\n * animationFrameScheduler.schedule(function(height) {\n * div.style.height = height + \"px\";\n *\n * this.schedule(height + 1); // `this` references currently executing Action,\n * // which we reschedule with new state\n * }, 0, 0);\n *\n * // You will see a div element growing in height\n * ```\n */\n\nexport const animationFrameScheduler = new AnimationFrameScheduler(AnimationFrameAction);\n\n/**\n * @deprecated Renamed to {@link animationFrameScheduler}. Will be removed in v8.\n */\nexport const animationFrame = animationFrameScheduler;\n", "import { Observable } from '../Observable';\nimport { SchedulerLike } from '../types';\n\n/**\n * A simple Observable that emits no items to the Observer and immediately\n * emits a complete notification.\n *\n * Just emits 'complete', and nothing else.\n *\n * ![](empty.png)\n *\n * A simple Observable that only emits the complete notification. It can be used\n * for composing with other Observables, such as in a {@link mergeMap}.\n *\n * ## Examples\n *\n * Log complete notification\n *\n * ```ts\n * import { EMPTY } from 'rxjs';\n *\n * EMPTY.subscribe({\n * next: () => console.log('Next'),\n * complete: () => console.log('Complete!')\n * });\n *\n * // Outputs\n * // Complete!\n * ```\n *\n * Emit the number 7, then complete\n *\n * ```ts\n * import { EMPTY, startWith } from 'rxjs';\n *\n * const result = EMPTY.pipe(startWith(7));\n * result.subscribe(x => console.log(x));\n *\n * // Outputs\n * // 7\n * ```\n *\n * Map and flatten only odd numbers to the sequence `'a'`, `'b'`, `'c'`\n *\n * ```ts\n * import { interval, mergeMap, of, EMPTY } from 'rxjs';\n *\n * const interval$ = interval(1000);\n * const result = interval$.pipe(\n * mergeMap(x => x % 2 === 1 ? of('a', 'b', 'c') : EMPTY),\n * );\n * result.subscribe(x => console.log(x));\n *\n * // Results in the following to the console:\n * // x is equal to the count on the interval, e.g. (0, 1, 2, 3, ...)\n * // x will occur every 1000ms\n * // if x % 2 is equal to 1, print a, b, c (each on its own)\n * // if x % 2 is not equal to 1, nothing will be output\n * ```\n *\n * @see {@link Observable}\n * @see {@link NEVER}\n * @see {@link of}\n * @see {@link throwError}\n */\nexport const EMPTY = new Observable((subscriber) => subscriber.complete());\n\n/**\n * @param scheduler A {@link SchedulerLike} to use for scheduling\n * the emission of the complete notification.\n * @deprecated Replaced with the {@link EMPTY} constant or {@link scheduled} (e.g. `scheduled([], scheduler)`). Will be removed in v8.\n */\nexport function empty(scheduler?: SchedulerLike) {\n return scheduler ? emptyScheduled(scheduler) : EMPTY;\n}\n\nfunction emptyScheduled(scheduler: SchedulerLike) {\n return new Observable((subscriber) => scheduler.schedule(() => subscriber.complete()));\n}\n", "import { SchedulerLike } from '../types';\nimport { isFunction } from './isFunction';\n\nexport function isScheduler(value: any): value is SchedulerLike {\n return value && isFunction(value.schedule);\n}\n", "import { SchedulerLike } from '../types';\nimport { isFunction } from './isFunction';\nimport { isScheduler } from './isScheduler';\n\nfunction last(arr: T[]): T | undefined {\n return arr[arr.length - 1];\n}\n\nexport function popResultSelector(args: any[]): ((...args: unknown[]) => unknown) | undefined {\n return isFunction(last(args)) ? args.pop() : undefined;\n}\n\nexport function popScheduler(args: any[]): SchedulerLike | undefined {\n return isScheduler(last(args)) ? args.pop() : undefined;\n}\n\nexport function popNumber(args: any[], defaultValue: number): number {\n return typeof last(args) === 'number' ? args.pop()! : defaultValue;\n}\n", "export const isArrayLike = ((x: any): x is ArrayLike => x && typeof x.length === 'number' && typeof x !== 'function');", "import { isFunction } from \"./isFunction\";\n\n/**\n * Tests to see if the object is \"thennable\".\n * @param value the object to test\n */\nexport function isPromise(value: any): value is PromiseLike {\n return isFunction(value?.then);\n}\n", "import { InteropObservable } from '../types';\nimport { observable as Symbol_observable } from '../symbol/observable';\nimport { isFunction } from './isFunction';\n\n/** Identifies an input as being Observable (but not necessary an Rx Observable) */\nexport function isInteropObservable(input: any): input is InteropObservable {\n return isFunction(input[Symbol_observable]);\n}\n", "import { isFunction } from './isFunction';\n\nexport function isAsyncIterable(obj: any): obj is AsyncIterable {\n return Symbol.asyncIterator && isFunction(obj?.[Symbol.asyncIterator]);\n}\n", "/**\n * Creates the TypeError to throw if an invalid object is passed to `from` or `scheduled`.\n * @param input The object that was passed.\n */\nexport function createInvalidObservableTypeError(input: any) {\n // TODO: We should create error codes that can be looked up, so this can be less verbose.\n return new TypeError(\n `You provided ${\n input !== null && typeof input === 'object' ? 'an invalid object' : `'${input}'`\n } where a stream was expected. You can provide an Observable, Promise, ReadableStream, Array, AsyncIterable, or Iterable.`\n );\n}\n", "export function getSymbolIterator(): symbol {\n if (typeof Symbol !== 'function' || !Symbol.iterator) {\n return '@@iterator' as any;\n }\n\n return Symbol.iterator;\n}\n\nexport const iterator = getSymbolIterator();\n", "import { iterator as Symbol_iterator } from '../symbol/iterator';\nimport { isFunction } from './isFunction';\n\n/** Identifies an input as being an Iterable */\nexport function isIterable(input: any): input is Iterable {\n return isFunction(input?.[Symbol_iterator]);\n}\n", "import { ReadableStreamLike } from '../types';\nimport { isFunction } from './isFunction';\n\nexport async function* readableStreamLikeToAsyncGenerator(readableStream: ReadableStreamLike): AsyncGenerator {\n const reader = readableStream.getReader();\n try {\n while (true) {\n const { value, done } = await reader.read();\n if (done) {\n return;\n }\n yield value!;\n }\n } finally {\n reader.releaseLock();\n }\n}\n\nexport function isReadableStreamLike(obj: any): obj is ReadableStreamLike {\n // We don't want to use instanceof checks because they would return\n // false for instances from another Realm, like an