1. ホーム
  2. angularjs

[解決済み】AngularJSでディレクティブを書くとき、新しいスコープ、新しい子スコープ、または新しい分離されたスコープが必要であるかどうかは、どのように決定するのですか?

2022-04-20 17:29:59

質問

新しいディレクティブを書くときに、どのタイプのスコープを使うべきかを判断するのに役立つガイドラインを探しています。理想を言えば、フローチャートのようなもので、いくつかの質問に答えていくと、正しい答え(新しい新しいスコープ、新しい子スコープ、新しいアイソレートスコープ)が出てくるようなものがいいのですが、それはあまりに無理な注文でしょう。以下は、私の現在のわずかなガイドラインです。

ある要素で孤立したスコープを持つディレクティブを使用すると、その同じ要素上の他のすべてのディレクティブが同じ(1つの)孤立したスコープを使用することを強制されることは承知していますが、これは孤立したスコープが使用できる場合を著しく制限しませんか。

Angular-UIチーム(または多くのディレクティブを書いたことのある他の人)が、その経験を共有してくれることを期待しています。

単に "再利用可能なコンポーネントのために分離されたスコープを使用する" と言う答えを追加しないでください。

どのように解決するのですか?

なんという素晴らしい質問でしょう。私は 他の人の意見も聞いてみたいですが、私が使っているガイドラインは以下の通りです。

高所作業前提:スコープは、親コントローラ、ディレクティブ、ディレクティブテンプレートの間の通信に使う "glue" として使用されます。

親スコープのこと。 scope: false ということで、新しいスコープは全くありません。

私はこの方法をあまり使いませんが、@MarkRajcok が言ったように、ディレクティブがスコープ変数にアクセスしない(そして明らかに何も設定しない!)なら、私に関する限り、これはちょうどいいのです。これはまた、子ディレクティブが のみ 親ディレクティブのコンテキストで使用され (ただし、これには常に例外があります)、テンプレートがないもの。基本的に、テンプレートを持つものはスコープを共有することには適しません。なぜなら、アクセスや操作のためにスコープを公開することになるからです(ただし、このルールには例外があると思います)。

例として、私は最近、私が書きかけのSVGライブラリを使って(静的な)ベクターグラフィックを描くディレクティブを作成しました。それは $observe の2つの属性( widthheight を使用しますが、スコープ変数の設定も読み込みもせず、テンプレートも持ちません。これは、もうひとつのスコープを作らないという意味では、良い使用例です。

しかし、別のSVGディレクティブでは、使用するデータセットが必要で、さらに、ほんの少しの状態を保存する必要がありました。この場合、親スコープを使うのは無責任です(これも一般論ですが)。そこで、代わりに...

子スコープです。 scope: true

子スコープを持つディレクティブはコンテキストを認識し、現在のスコープと対話することを意図しています。

明らかに、分離型スコープと比較した場合の主な利点は、ユーザーが望む属性に自由に補間を使用できることです。 class="item-type-{{item.type}}" はデフォルトでは動作しませんが、子スコープを持つディレクティブでは問題なく動作します。なぜなら、補間されるものはデフォルトで親スコープに存在するからです。また、ディレクティブ自身は、親の汚染を心配することなく、自分のスコープのコンテキストで安全に属性や式を評価することができます。

例えば、ツールチップはただ追加されるものです。他のディレクティブや補間された属性をここで使用することが予想されるため、分離スコープは機能しません(デフォルトで、以下を参照ください)。ツールチップは単なる機能拡張です。しかし、tooltip はサブディレクティブやテンプレートで使用するためにスコープにいくつかのものを設定する必要があり、明らかにそれ自身の状態を管理するため、親スコープを使用することは非常に悪いことです。そのため、親スコープを使用することは非常に悪いことです。

私は、孤立スコープや親スコープよりも、子スコープを使うことが多いと感じています。

アイソレートスコープ。 scope: {}

これは再利用可能なコンポーネントのためのものです :-)

でも、真面目な話、quot;再利用可能なコンポーネントとは、quot;自己完結型のコンポーネントだと考えています。その意図は、特定の目的のために使用されることであり、他のディレクティブと組み合わせたり、DOMノードに他の補間属性を追加することは、本来は意味を持ちません。

具体的には、このスタンドアロン機能に必要なものは、親スコープのコンテキストで評価される指定された属性を通じて提供されます。これらは、一方向の文字列('@')、一方向の式('&')、または双方向の変数結合('=')となります。

自己完結型のコンポーネントでは、それ自体が存在するため、他のディレクティブや属性を適用する必要がないのが特徴です。そのスタイルは、(必要であれば)それ自身のテンプレートによって管理され、(必要であれば)適切なコンテンツをトランスクルードさせることができます。これはスタンドアロンなので、私たちはこれを孤立したスコープに置いて、次のように言っています: "これをいじってはいけません。私は、これらのいくつかの属性を通じて、定義されたAPIを提供します。

良いベストプラクティスは、ディレクティブリンクとコントローラ関数から、テンプレートベースのものをできるだけ除外することです。これは、もうひとつの API ライクな設定ポイントを提供します。機能はすべて同じままで、その内部APIは決して触られていませんが、必要なだけスタイルとDOMの実装をいじることができます。ui/bootstrapは Peter & Pawelは素晴らしいので、これをうまくやる方法の例です。

アイソレート・スコープもトランスクルージョンと一緒に使うのに最適です。タブを例にとると、機能全体だけでなく、何であれ 内部 タブ(とペイン)に好きなことをさせながら、その親スコープから自由に評価することができるのです。タブは明らかに自分の 状態 しかし、その状態は、それが使われたコンテキストとは関係なく、タブディレクティブをタブディレクティブたらしめているのは、完全に内部的なものなのです。さらに、他のディレクティブをタブに使用することはあまり意味がありません。タブはタブであり、その機能はすでに持っているのですから。

より多くの機能で囲むか、より多くの機能を転記するか、しかしディレクティブはすでにあるものなのです。

とはいえ、@ProLoser氏の回答で示唆されているように、アイソレートスコープの制限(=特徴)のいくつかを回避する方法があることに留意しておく必要があります。例えば、子スコープセクションで、(デフォルトで)アイソレートスコープを使用すると、非ディレクティブ属性の補間が壊れると書きました。しかし、ユーザーは、例えば、単純に class="item-type-{{$parent.item.type}}" で、再び動作するようになります。ですから、もし子スコープよりも分離スコープを使うべきやむを得ない理由があるけれども、これらの制限のいくつかを心配しているのなら、必要なら事実上すべての制限を回避することができることを知っておいてください。

概要

新しいスコープを持たないディレクティブは読み取り専用で、完全に信頼され(つまりアプリの内部で)、ジャックに触れることはありません。子スコープを持つディレクティブ 追加 の機能を持ちますが、それらは 唯一の 機能です。最後に、分離スコープは、ゴール全体であるディレクティブのためのものです。それらはスタンドアロンなので、それらを不正に動作させても問題ありません(そして最も"正しい")。

最初に考えたことを吐き出したかったのですが、またいろいろ思いついたら、更新します。しかし、なんてこった......SOの回答にしては長いな......。


PS: 全く余談ですが、スコープについて話しているので、私は"prototypical"と言うのが好きなのですが、他の人は"prototypal"を好み、より正確に見えるけど舌触りは全く良くないですね :-)。