스타일 쿼리 시작하기

상위 요소의 인라인 크기 및 컨테이너 검색어 단위 값을 쿼리하는 기능이 최근 모든 최신 브라우저 엔진에서 안정적으로 지원됩니다.

브라우저 지원

  • 105
  • 105
  • 110
  • 16

소스

하지만 포함 사양에는 크기 쿼리 외에도 많은 항목이 포함되어 있습니다. 또한 상위 요소의 스타일 값을 쿼리할 수도 있습니다. Chromium 111부터 맞춤 속성 값에 스타일 포함을 적용하고 상위 요소에서 맞춤 속성 값을 쿼리할 수 있습니다.

브라우저 지원

  • 111
  • 111
  • x
  • x

소스

즉, CSS에서 스타일을 훨씬 더 논리적으로 제어할 수 있으며 애플리케이션 로직과 데이터 레이어를 스타일에서 더 잘 구분할 수 있습니다.

크기 및 스타일 쿼리를 다루는 CSS Containment Module 레벨 3 사양으로 font-weight: 800와 같은 속성 및 값 쌍을 포함하여 상위 요소에서 스타일을 쿼리할 수 있습니다. 하지만 이 기능이 출시될 때 스타일 쿼리는 현재 CSS 맞춤 속성 값에만 사용할 수 있습니다. 이는 스타일을 결합하고 디자인과 데이터를 구분하는 데 여전히 매우 유용합니다. CSS 맞춤 속성이 있는 스타일 쿼리를 사용하는 방법을 살펴보겠습니다.

스타일 쿼리 시작하기

다음 HTML이 있다고 가정해 보겠습니다.

<ul class="card-list">
  <li class="card-container">
    <div class="card">
      ...
    </div>
  </li>
</ul>

스타일 쿼리를 사용하려면 먼저 컨테이너 요소를 설정해야 합니다. 이를 위해서는 직접 상위 또는 간접 상위 요소를 쿼리하는지에 따라 약간 다른 접근 방식이 필요합니다.

직계 상위 쿼리

스타일 쿼리의 다이어그램

스타일 쿼리와 달리 container-type 또는 container 속성을 사용하여 포함을 .card-container에 적용하지 않아도 .card에서 직접 상위 요소의 스타일을 쿼리할 수 있습니다. 그러나 컨테이너 (이 경우 .card-container) 또는 DOM에서 스타일을 지정할 요소를 포함하는 요소에 스타일 (이 경우 맞춤 속성 값)을 적용해야 합니다. 해당 쿼리를 사용하여 스타일을 지정하는 직접 요소에 쿼리하는 스타일을 적용할 수 없습니다. 무한 루프가 발생할 수 있기 때문입니다.

상위 요소에 직접 쿼리하려면 다음과 같이 작성하면 됩니다.

/* styling .card based on the value of --theme on .card-container */
@container style(--theme: warm) {
  .card {
    background-color: wheat;
    border-color: brown; 
    ...
  }
}

스타일 쿼리는 쿼리를 style()로 래핑하는 것을 볼 수 있습니다. 이는 스타일과 크기 값을 구분하기 위함입니다. 예를 들어 컨테이너 너비에 관한 쿼리를 @container (min-width: 200px) { … }로 작성할 수 있습니다. 이렇게 하면 상위 컨테이너의 너비가 200픽셀 이상일 때 스타일이 적용됩니다. 그러나 min-width는 CSS 속성이 될 수도 있으며 스타일 쿼리를 사용하여 min-width의 CSS 값을 쿼리할 수 있습니다. 따라서 style() 래퍼를 사용하여 차이를 명확히 하는 것이 좋습니다(@container style(min-width: 200px) { … }).

간접 상위 요소 스타일 지정

직접 상위 요소가 아닌 요소의 스타일을 쿼리하려면 해당 요소에 container-name을 부여해야 합니다. 예를 들어 .card-listcontainer-name를 제공하고 스타일 쿼리에서 이를 참조하여 .card-list의 스타일을 기반으로 .card에 스타일을 적용할 수 있습니다.

/* styling .card based on the value of --moreGlobalVar on .card-list */
@container cards style(--moreGlobalVar: value) {
  .card {
    ...
  }
}

일반적으로 쿼리하는 내용을 명확히 하고 이러한 컨테이너에 더 쉽게 액세스할 수 있도록 컨테이너 이름을 지정하는 것이 좋습니다. 이 방법이 유용한 한 가지 예는 .card 내에서 직접 요소의 스타일을 지정하려는 경우입니다. .card-container에 이름이 지정된 컨테이너가 없으면 직접 쿼리할 수 없습니다.

하지만 이 모든 것이 실제로는 훨씬 더 합리적입니다. 몇 가지 예를 살펴보겠습니다.

쿼리 스타일 지정

여러 제품 카드가 있는 데모 이미지. 일부 제품에는 &#39;new&#39; 또는 &#39;low stock&#39; 태그가 있고 빨간색 배경에 &#39;낮은 재고&#39; 카드도 있습니다.

스타일 쿼리는 여러 변형이 있는 재사용 가능한 구성요소가 있거나 모든 스타일을 제어할 수는 없지만 특정 경우에 변경사항을 적용해야 하는 경우 특히 유용합니다. 이 예는 동일한 카드 구성요소를 공유하는 제품 카드 집합을 보여줍니다. 일부 제품 카드에는 --detail이라는 맞춤 속성에 의해 트리거되는 '신규' 또는 '낮은 재고'와 같은 추가 세부정보/메모가 있습니다. 또한 제품이 '낮은 재고' 상태인 경우 진한 빨간색 테두리 배경이 표시됩니다. 이러한 유형의 정보는 서버에서 렌더링될 가능성이 높으며 다음과 같이 인라인 스타일을 통해 카드에 적용될 수 있습니다.

 <div class="product-list">
  <div class="product-card-container" style="--detail: new">
    <div class="product-card">
      <div class="media">
        <img .../>
      <div class="comment-block"></div>
    </div>
  </div>
  <div class="meta">
    ...
  </div>
  </div>
  <div class="product-card-container" style="--detail: low-stock">
    ...
  </div>
  <div class="product-card-container">
    ...
  </div>
  ...
</div>

이 구조화된 데이터가 주어지면 --detail에 값을 전달하고 이 CSS 맞춤 속성을 사용하여 스타일을 적용할 수 있습니다.

@container style(--detail: new) {
  .comment-block {
    display: block;
  }
  
  .comment-block::after {
    content: 'New';
    border: 1px solid currentColor;
    background: white;
    ...
  }
}

@container style(--detail: low-stock) {
  .comment-block {
    display: block;
  }
  
  .comment-block::after {
    content: 'Low Stock';
    border: 1px solid currentColor;
    background: white;
    ...
  }
  
  .media-img {
    border: 2px solid brickred;
  }
}

위의 코드를 사용하면 --detail: low-stock--detail: new용 칩을 적용할 수 있지만 코드 블록에서 일부 중복성을 발견할 수도 있습니다. 현재 @container style(--detail)--detail의 존재만 쿼리할 수 있는 방법은 없습니다. 이를 통해 스타일을 더 효과적으로 공유하고 반복을 줄일 수 있습니다. 이 기능은 현재 작업 그룹에서 토론 중입니다.

날씨 카드

이전 예에서는 여러 개의 가능한 값을 가진 단일 맞춤 속성을 사용하여 스타일을 적용했습니다. 하지만 여러 커스텀 속성을 사용하고 쿼리하는 방법으로 혼합할 수도 있습니다. 이 날씨 카드의 예를 살펴보겠습니다.

날씨 카드 데모

이러한 카드의 배경 그라데이션 및 아이콘의 스타일을 지정하려면 '흐림', '비' 또는 '맑음'과 같은 날씨 특성을 확인하세요.

@container style(--sunny: true) {
  .weather-card {
    background: linear-gradient(-30deg, yellow, orange);
  }
  
  .weather-card:after {
    content: url(<data-uri-for-demo-brevity>);
    background: gold;
  }
}

이렇게 하면 고유한 특성에 따라 각 카드의 스타일을 지정할 수 있습니다. 하지만 미디어 쿼리와 동일한 방식으로 and 조합을 사용하여 특징 (맞춤 속성) 조합의 스타일을 지정할 수도 있습니다. 예를 들어 흐리고 맑은 날은 다음과 같습니다.

@container style(--sunny: true) and style(--cloudy: true) {
    .weather-card {
      background: linear-gradient(24deg, pink, violet);
    }
  
  .weather-card:after {
      content: url(<data-uri-for-demo-brevity>);
      background: violet;
  }
}

설계에서 데이터 분리

두 데모 모두 적용된 스타일에서 데이터 영역 (페이지에서 렌더링될 DOM)을 분리할 수 있다는 구조적 이점이 있습니다. 스타일은 구성요소 스타일 내에 있는 가능한 변형으로 작성되는 반면, 엔드포인트는 구성요소의 스타일을 지정하는 데 사용할 데이터를 전송할 수 있습니다. 첫 번째 사례에서 --detail 값을 업데이트하는 경우와 같이 단일 값을 사용하거나 두 번째 경우 (--rainy 또는 --cloudy 또는 --sunny를 설정)와 같이 여러 변수를 사용할 수 있습니다. 가장 좋은 점은 이러한 값도 결합할 수 있다는 것입니다. --sunny--cloudy를 모두 확인하면 부분적으로 흐린 스타일이 표시될 수 있습니다.

자바스크립트를 통해 맞춤 속성 값을 업데이트하는 작업은 DOM 모델을 설정하는 동안 (즉, 프레임워크에서 구성요소를 빌드하는 동안)하거나 <parentElem>.style.setProperty('--myProperty’, <value>)를 사용하여 언제든지 업데이트할 수 있습니다. I

다음은 몇 줄의 코드로 버튼의 --theme를 업데이트하고 스타일 쿼리와 맞춤 속성 (--theme)을 사용하여 스타일을 적용하는 데모입니다.

스타일 쿼리를 사용하여 카드의 스타일을 지정합니다. 맞춤 속성 값을 업데이트하는 데 사용되는 자바스크립트는 다음과 같습니다.

const themePicker = document.querySelector('#theme-picker')
const btnParent = document.querySelector('.btn-section');

themePicker.addEventListener('input', (e) => {
  btnParent.style.setProperty('--theme', e.target.value);
})

이 도움말에서 설명하는 기능은 시작에 불과합니다. 컨테이너 쿼리에서 동적 반응형 인터페이스를 빌드하는 데 도움이 되는 더 많은 기능을 기대할 수 있습니다. 특히 스타일 쿼리에 관해서는 아직 해결되지 않은 몇 가지 문제가 있습니다. 그 중 하나는 맞춤 속성 이외에 CSS 스타일에 대한 스타일 쿼리를 구현하는 것입니다. 이 기능은 이미 현재 사양 수준에 포함되어 있지만 아직 어떤 브라우저에서도 구현되지 않았습니다. 불리언 컨텍스트 평가는 해결되지 않은 문제가 해결되면 현재 사양 수준에 추가될 것으로 예상되며, 범위 쿼리는 다음 사양 수준을 위해 계획됩니다.