Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Angular2 Styles in a Directive

In the given examples of Attribute directives (i.e. a directive to add appearance/behaviour), we have a fairly simple setting of a style on the host element.. e.g.

import {Directive, ElementRef } from 'angular2/core'; @Directive({     selector: '[myHighlight]' }) export class HighlightDirective {     constructor(element) {        element.nativeElement.style.backgroundColor = 'yellow';     }  static get parameters(){     return [[ElementRef]]; } 

Rather than setting the style, can i use a styles instead? e.g.

@Directive({     selector: '[myHighlight]',     styles: [':host { background-color: yellow; }'] }) 

This doesn't seem to work for me?

I'm doing something slightly more complex which has led to a fair amount of monolothic code, setting lots of styles, using AnimationBuilder etc etc. feels to me like it would be much better to seperate this out into classes and animations in a CSS.

ViewEncapsulation = emulated/default if that matters?

like image 455
ct5845 Avatar asked Mar 10 '16 11:03

ct5845


People also ask

Can a directive have styles?

You can use a css injection service to facilitate directives that want to bring their own styles. This allows you to keep any resource declarations close to the directive definition.

Which directive is used to set style?

The NgStyle directive lets you set a given DOM elements style properties. This sets the background color of the div to green. The above code uses the ternary operator to set the background color to green if the persons country is the UK else red.

How many types of directives exist in angular7?

Angular directives can be classified into three types: Component Directives: It forms the main class and is declared by @Component.


2 Answers

You can use host binding to bind to style attributes:

@Directive({     selector: '[myHighlight]',     host: {       '[style.background-color]': '"yellow"',     } }) 

or

@Directive({     selector: '[myHighlight]', }) class MyDirective {   @HostBinding('style.background-color')   backgroundColor:string = 'yellow'; } 
like image 166
Günter Zöchbauer Avatar answered Oct 05 '22 20:10

Günter Zöchbauer


While the other answers are helpful in most circumstances, you seem to need a more traditional CSS stylesheet approach like I had a use case for.

The issue is Angular's default of emulating a Shadow DOM which scopes styles only within the host element.

TWO OPTIONS:

1)

You can tell Angular to cascade your styles down through all its descendants using the :host /deep/ .some-style-to-cascade-down-like-normal {} or replace /deep/ with >>>. See Angular's Docs about this.

Three important things to note:

  • ViewEncapsulation needs to be its default (emulated) state
  • Angular/Chrome are deprecating both these syntaxes while they are working on a better approach
  • If you're using the Angular CLI, you have to use the /deep/ instead of >>>

2)

Though you'll loose the scoped component encapsulation (if that matters in your case), here is an example using "myHighlight" as a directive though TypeScripted as a component so I can import the stylesheet:

USAGE:
<p myHighlight>Highlight me!</p>

TS (component treated as a directive):

import {     Component,     ViewEncapsulation } from '@angular/core';  @Component({     selector: 'p[myHighlight]', // Refer to it like an attribute directive     templateUrl: './my-highlight.component.html',     styleUrls: ['./my-highlight.component.scss'],     encapsulation: ViewEncapsulation.None // Tell Angular to not scope your styles }) 

Angular Material 2's Button uses this same approach to solve this issue.

And here's a great article called All the Ways to Add CSS to Angular 2 Components which brought me to this awareness and explains how Angular treats all three ViewEncapsulation properties.

like image 31
Modular Avatar answered Oct 05 '22 22:10

Modular