2
5

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

【Angular】コンポーネント要素にclassを付与するやり方

Last updated at Posted at 2020-06-07

はじめに

Angularではhtmlをコンポーネントに切り出すと、作成したコンポーネントも一つのDOMとなります。
そのためコンポーネントに切り出す前と比べてDOMの階層が1階層増えてしまいます。このときに起こる問題として、例えばDOMの親子関係を前提としたCSSを適用させていた場合に元のDOMの階層と変わってしまうがために親子関係が崩れ、適用したいCSSのスタイルが当たらなくなってしまうことがあります。
ここでは切り出したコンポーネント要素自体にclassや属性を付与することでそれを回避した方法を記します。

前提

Angular 9.1.0を前提としています。今後のバージョン次第で記述方法や動作が変わる可能性があります。

具体的にどんなときに起こる問題か

例えば以下のコードがあるとします。

sample.html
<div class="form">
  <input type="text"/>
  <div class="select">
    <select>
      <option>A</option>
      <option>B</option>
      <option>C</option>
    </select>
  </div>
  <button>追加</button>
</div>
sample.css
.form > .select {
  padding: 5px;
  width: 100%;
}

CSSではformクラスの直下にあるselectクラスのpaddingが5px, 横幅が100%になるように記述しています。

上記の場合、生成されるDOM階層は以下のようになります。

div(.form)
 |-- input
 |-- div(.select)
   |-- select
     |-- option
     |-- option
     |-- option
 |-button

このとき、selectクラスのdiv要素部分をコンポーネントに切り出すとします。

select.component.html
<div class="select">
  <select>
    <option>A</option>
    <option>B</option>
    <option>C</option>
  </select>
</div>
select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: []
})
export class SelectComponent {
}
sample.html
<div class="form">
  <input type="text"/>
  <app-select></app-select>
  <button>追加</button>
</div>

上記のようにapp-selectというコンポーネントに切り出しました。
この場合生成されるDOMは以下のような階層構造になります。

div(.form)
 |-- input
 |-- app-select
   |-- div(.select)
     |-- select
       |-- option
       |-- option
       |-- option
 |-button

元々の階層構造ではdiv.form > div.selectという構造でしたが、コンポーネントに切り出したことで元のDOMの階層構造と変わってしまい、div.form > app-select > div.selectという構造になります。
今回のように親子関係のDOMを前提としたCSSがある場合にそれを崩すようなコンポーネントの切り方をしてしまうとCSSスタイルが適用されず、表示が崩れてしまう現象が起こってしまいます。

どう解決するか

解決方法としてはいくつかあるかと思いますが、今回はなるべくCSSを触らずに回避したいと思います。(エンジニアはあまり触りたくないですよね?(偏見))
app-select要素自体にクラスを適用できれば.form > .selectの親子関係は維持できるため、CSSスタイルを適用できそうです。
今回はその方法で回避します。

つまり、この記事ではコンポーネント要素自体にクラスを付与させる方法を紹介します。
静的にクラスを付与させる方法と動的にクラスを付与させる方法の2通りを紹介します。

コンポーネント要素に静的にクラスを付与させる方法

@Componentにはhostオプションを定義することができます。
hostオプションは@Directiveアノテーションから継承しており、プロパティや属性、イベントをkey-valueの形式で与えることができます。

host: {
    [key: string]: string;
}

例えば先程のapp-selectにクラスを付与する場合、以下のように書くことができます。

select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: [],
  host: {
    class: 'select' // 付与したいclassをスペース区切りで指定する
  }
})
export class SelectComponent {
}

これにより、app-selectにselectクラスを付与することができました。

コンポーネントルートに動的にクラスを付与させる方法

@HostBindingアノテーションを利用します。役割は先程の@Componentのhostオプションとほぼ同じです。
@HostBindingはhostPropertyNameオプションだけを持ちます。hostPropertyNameにはコンポーネント要素に付与したい属性を指定できます。
指定の仕方は<属性>.<属性名>で指定します。

例えば先程のapp-selectにクラスを付与する場合、以下のように書くことができます。

select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: []
})
export class SelectComponent {
  @HostBinding('class.select') selectClass: boolean = true;  // trueの場合付与される
}

@HostBindingは変数に付与できるため、クラスの付け外しを以下のように動的に行うことができます。

select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: []
})
export class SelectComponent {
  @HostBinding('class.select') selectClass: boolean = true;

  @Input
  set select(isSelect: boolean) {
    this.selectClass = isSelect;  // 親から渡ってくるisSelectによってapp-selectにselectクラスを付与させたりさせなかったりできる
  }
}

ちなみに@HostBindingを指定した変数をreadonlyとすれば静的に付与させることも可能です。

select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: []
})
export class SelectComponent {
  @HostBinding('class.select') readonly selectClass: boolean = true; // 常にselectクラスを適用する
}

まとめ

ということでコンポーネントに切り出したときにCSSスタイルの崩れが起こりうる現象とその解決方方法としてコンポーネント要素にクラスを付与する方法を紹介しました。
コンポーネント要素に属性を付与したい場合は

  • 静的に付与したい→@Componentのhostプロパティか@HostBindingで変数をreadonlyにして指定
  • 動的に付与したい→@HostBindingで指定

で指定しましょう。

おまけ:コンポーネント要素自体にCSSスタイルを適用する方法

クラスを付与するのではなく直接CSSを指定したい場合、:hostセレクタを使用してCSSスタイルを適用することができます。

select.component.html
<div class="select">
 <select>
    <option>A</option>
    <option>B</option>
    <option>C</option>
  </select>
</div>
select.component.ts
@Component({
  selector: 'app-select',
  templateUrl: './select.component.html',
  styleUrls: ['./select.component.scss']
})
export class SelectComponent {
}
select.component.scss
:host { // :hostセレクタでコンポーネント要素自体に適用可能
  padding: 5px;
  width: 100%;
}

その他参考

2
5
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
5

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?