จำกัดการเข้าถึงของตัวเลือกด้วยกฎ @scope ของ CSS

ดูวิธีใช้ @scope เพื่อเลือกองค์ประกอบภายในซับต้นไม้ที่จำกัดของ DOM เท่านั้น

การรองรับเบราว์เซอร์

  • Chrome: 118
  • Edge: 118
  • Firefox: อยู่หลังธง
  • Safari: 17.4

แหล่งที่มา

ศิลปะการเขียนตัวเลือก CSS ที่ละเอียดอ่อน

เมื่อเขียนตัวเลือก คุณอาจพบว่าตัวเองต้องเลือกระหว่าง 2 โลก ในทางหนึ่ง คุณควรเลือกองค์ประกอบให้เจาะจง ในทางกลับกัน คุณก็ต้องการให้ตัวเลือกยังคงลบล้างได้ง่ายและไม่ได้เชื่อมโยงกับโครงสร้าง DOM อย่างแน่นหนา

เช่น เมื่อคุณต้องการเลือก "รูปภาพหลักในพื้นที่เนื้อหาของคอมโพเนนต์การ์ด" ซึ่งเป็นการเลือกองค์ประกอบที่ค่อนข้างเฉพาะเจาะจง คุณอาจไม่ต้องการเขียนตัวเลือกอย่าง .card > .content > img.hero

  • ตัวเลือกนี้มีความเฉพาะเจาะจงสูงถึง (0,3,1) ซึ่งทําให้การลบล้างยากขึ้นเมื่อโค้ดมีความยาวมากขึ้น
  • การใช้คอมบิเนเตอร์รายการย่อยโดยตรงจะเชื่อมโยงกับโครงสร้าง DOM อย่างแน่นหนา หากมาร์กอัปมีการเปลี่ยนแปลง คุณจะต้องเปลี่ยน CSS ด้วย

แต่คุณก็ไม่ต้องการเขียนแค่ img เป็นตัวเลือกสําหรับองค์ประกอบนั้น เนื่องจากจะเลือกองค์ประกอบรูปภาพทั้งหมดในหน้าเว็บ

การหาสมดุลที่เหมาะสมในเรื่องนี้มักเป็นเรื่องท้าทาย ในช่วงหลายปีที่ผ่านมา นักพัฒนาแอปบางรายได้คิดค้นวิธีแก้ปัญหาและวิธีแก้ปัญหาชั่วคราวเพื่อช่วยคุณในสถานการณ์เช่นนี้ เช่น

  • วิธีการต่างๆ เช่น BEM กําหนดให้คุณกําหนดคลาส card__img card__img--hero ให้กับองค์ประกอบนั้นเพื่อลดความเฉพาะเจาะจง ในขณะที่ให้คุณเลือกองค์ประกอบที่เฉพาะเจาะจงได้
  • โซลูชันที่ใช้ JavaScript เช่น CSS แบบจำกัดขอบเขตหรือคอมโพเนนต์ที่มีสไตล์จะเขียนตัวเลือกทั้งหมดของคุณใหม่โดยการเพิ่มสตริงที่สร้างขึ้นแบบสุ่ม เช่น sc-596d7e0e-4 ลงในตัวเลือกเพื่อป้องกันไม่ให้ตัวเลือกกำหนดเป้าหมายองค์ประกอบที่อีกฝั่งหนึ่งของหน้า
  • บางไลบรารีถึงกับยกเลิกการใช้ตัวเลือกไปเลยและกำหนดให้คุณใส่ทริกเกอร์การจัดสไตล์ในมาร์กอัปโดยตรง

แต่จะเกิดอะไรขึ้นหากคุณไม่ต้องการสิ่งเหล่านี้ จะเกิดอะไรขึ้นหาก CSS มีวิธีให้คุณเลือกองค์ประกอบได้อย่างเฉพาะเจาะจงโดยไม่ต้องเขียนตัวเลือกที่เฉพาะเจาะจงสูงหรือตัวเลือกที่เชื่อมโยงกับ DOM อย่างแน่นหนา ด้วยเหตุนี้ @scope จึงเข้ามามีบทบาท โดยให้คุณเลือกองค์ประกอบได้ภายในซับต้นไม้ของ DOM เท่านั้น

ขอแนะนำ @scope

@scope ช่วยให้คุณจํากัดการเข้าถึงของตัวเลือกได้ ซึ่งทำได้โดยการตั้งค่า scoping root ซึ่งจะเป็นตัวกําหนดขอบเขตบนของต้นไม้ย่อยที่คุณต้องการกําหนดเป้าหมาย เมื่อตั้งค่ารูทการกําหนดขอบเขตแล้ว กฎสไตล์ที่มีชื่อเรียกว่ากฎสไตล์ที่กําหนดขอบเขตจะเลือกได้จากซับต้นไม้ที่จำกัดของ DOM เท่านั้น

เช่น หากต้องการกำหนดเป้าหมายเฉพาะองค์ประกอบ <img> ในคอมโพเนนต์ .card ให้ตั้งค่า .card เป็นรูทการกําหนดขอบเขตของ at-rule @scope

@scope (.card) {
    img {
        border-color: green;
    }
}

กฎรูปแบบที่มีขอบเขต img { … } จะเลือกได้เฉพาะองค์ประกอบ <img> ที่อยู่ในขอบเขตขององค์ประกอบ .card ที่ตรงกัน

หากไม่ต้องการให้ระบบเลือกองค์ประกอบ <img> ภายในพื้นที่เนื้อหาของการ์ด (.card__content) ให้ทําการเลือก img ให้เฉพาะเจาะจงมากขึ้น อีกวิธีหนึ่งคือใช้ประโยชน์จากข้อเท็จจริงที่ว่า @scope at-rule ยังยอมรับขีดจํากัดการกําหนดขอบเขตด้วย ซึ่งจะเป็นตัวกําหนดขอบเขตล่าง

@scope (.card) to (.card__content) {
    img {
        border-color: green;
    }
}

กฎรูปแบบที่มีขอบเขตนี้จะกำหนดเป้าหมายเฉพาะองค์ประกอบ <img> ที่วางไว้ระหว่างองค์ประกอบ .card กับ .card__content ในลําดับชั้นบรรพบุรุษ การกําหนดขอบเขตประเภทนี้ซึ่งมีขอบเขตบนและล่างมักเรียกว่าขอบเขตโดนัท

ตัวเลือก :scope

โดยค่าเริ่มต้น กฎรูปแบบที่มีขอบเขตทั้งหมดจะสัมพันธ์กับรูทการกําหนดขอบเขต นอกจากนี้ คุณยังกําหนดเป้าหมายองค์ประกอบรูทการกําหนดขอบเขตเองได้ด้วย ในกรณีนี้ ให้ใช้ตัวเลือก :scope

@scope (.card) {
    :scope {
        /* Selects the matched .card itself */
    }
    img {
       /* Selects img elements that are a child of .card */
    }
}

ระบบจะใส่ :scope ไว้ข้างหน้าตัวเลือกภายในกฎสไตล์ที่มีขอบเขตโดยปริยาย หากต้องการ คุณก็ระบุได้อย่างชัดเจนโดยใส่ :scope ไว้ข้างหน้า หรือจะใส่ตัวเลือก & ไว้ข้างหน้าก็ได้จากการฝัง CSS

@scope (.card) {
    img {
       /* Selects img elements that are a child of .card */
    }
    :scope img {
        /* Also selects img elements that are a child of .card */
    }
    & img {
        /* Also selects img elements that are a child of .card */
    }
}

ขีดจํากัดการกําหนดขอบเขตสามารถใช้คลาสจำลอง :scope เพื่อกำหนดความสัมพันธ์ที่เฉพาะเจาะจงกับรูทการกําหนดขอบเขตได้ ดังนี้

/* .content is only a limit when it is a direct child of the :scope */
@scope (.media-object) to (:scope > .content) { ... }

นอกจากนี้ ขีดจํากัดการกําหนดขอบเขตยังอ้างอิงองค์ประกอบที่อยู่นอกรูทการกําหนดขอบเขตได้ด้วยโดยใช้ :scope เช่น

/* .content is only a limit when the :scope is inside .sidebar */
@scope (.media-object) to (.sidebar :scope .content) { ... }

โปรดทราบว่ากฎสไตล์ที่มีขอบเขตจะหนีออกจากซับต้นไม้ไม่ได้ การเลือกอย่าง :scope + p ไม่ถูกต้องเนื่องจากพยายามเลือกองค์ประกอบที่ไม่ได้อยู่ในขอบเขต

@scope และความละเอียด

ตัวเลือกที่คุณใช้ในช่วงก่อนสําหรับ @scope จะไม่ส่งผลต่อความเฉพาะเจาะจงของตัวเลือกที่มี ในตัวอย่างนี้ ตัวเลือก img ยังคงมีความเฉพาะเจาะจงเป็น (0,0,1)

@scope (#sidebar) {
    img { /* Specificity = (0,0,1) */
        …
    }
}

ความเฉพาะเจาะจงของ :scope นั้นเหมือนกับของคลาสจำลองทั่วไป ซึ่งก็คือ (0,1,0)

@scope (#sidebar) {
    :scope img { /* Specificity = (0,1,0) + (0,0,1) = (0,1,1) */
        …
    }
}

ในตัวอย่างนี้ & จะได้รับการเปลี่ยนรูปแบบภายในเป็นตัวเลือกที่ใช้สำหรับรูทการกําหนดขอบเขต ซึ่งรวมอยู่ในตัวเลือก :is() สุดท้ายเบราว์เซอร์จะใช้ :is(#sidebar, .card) img เป็นตัวเลือกในการจับคู่ กระบวนการนี้เรียกว่าการกรองน้ำตาลออก

@scope (#sidebar, .card) {
    & img { /* desugars to `:is(#sidebar, .card) img` */
        …
    }
}

เนื่องจาก & ได้รับการกรองน้ำตาลออกโดยใช้ :is() ระบบจะคํานวณความเฉพาะเจาะจงของ & ตามกฎความเฉพาะเจาะจงของ :is() กล่าวคือ ความเฉพาะเจาะจงของ & จะเป็นความเฉพาะเจาะจงของอาร์กิวเมนต์ที่เฉพาะเจาะจงที่สุด

เมื่อนําไปใช้กับตัวอย่างนี้ ความเฉพาะเจาะจงของ :is(#sidebar, .card) คืออาร์กิวเมนต์ที่เฉพาะเจาะจงที่สุด ซึ่งก็คือ #sidebar และจึงกลายเป็น (1,0,0) รวมเข้ากับความเฉพาะเจาะจงของ img ซึ่งก็คือ (0,0,1) คุณจะได้ (1,0,1) ที่เป็นการเลือกที่เฉพาะเจาะจงสำหรับตัวเลือกที่ซับซ้อนทั้งหมด

@scope (#sidebar, .card) {
    & img { /* Specificity = (1,0,0) + (0,0,1) = (1,0,1) */
        …
    }
}

ความแตกต่างระหว่าง :scope กับ & ภายใน @scope

นอกจากความแตกต่างในวิธีคํานวณความเฉพาะเจาะจงแล้ว ความแตกต่างอีกอย่างหนึ่งระหว่าง :scope กับ & คือ :scope แสดงถึงรูทการกําหนดขอบเขตที่ตรงกัน ขณะที่ & แสดงถึงตัวเลือกที่ใช้จับคู่รูทการกําหนดขอบเขต

ด้วยเหตุนี้ คุณจึงใช้ & ได้หลายครั้ง ซึ่งต่างจาก :scope ที่คุณใช้ได้เพียงครั้งเดียว เนื่องจากคุณจับคู่รูทการกําหนดขอบเขตภายในรูทการกําหนดขอบเขตไม่ได้

@scope (.card) {
  & & { /* Selects a `.card` in the matched root .card */
  }
  :scope :scope { /* ❌ Does not work */
    …
  }
}

ขอบเขตที่ไม่มีช่วงเกริ่นนำ

เมื่อเขียนรูปแบบในบรรทัดด้วยองค์ประกอบ <style> คุณสามารถกําหนดขอบเขตกฎรูปแบบให้กับองค์ประกอบหลักที่ล้อมรอบองค์ประกอบ <style> ได้โดยไม่ได้ระบุรูทการกําหนดขอบเขต โดยตัดช่วงเกริ่นนำของ @scope ออก

<div class="card">
  <div class="card__header">
    <style>
      @scope {
        img {
          border-color: green;
        }
      }
    </style>
    <h1>Card Title</h1>
    <img src="…" height="32" class="hero">
  </div>
  <div class="card__content">
    <p><img src="…" height="32"></p>
  </div>
</div>

ในตัวอย่างข้างต้น กฎที่มีขอบเขตจะกำหนดเป้าหมายเฉพาะองค์ประกอบภายใน div ที่มีชื่อคลาส card__header เนื่องจาก div นั้นเป็นองค์ประกอบหลักขององค์ประกอบ <style>

@scope ในการแสดงผลตามลำดับขั้น

ภายใน CSS Cascade @scope ยังเพิ่มเกณฑ์ใหม่ด้วย ได้แก่ ความใกล้เคียงของการกําหนดขอบเขต ขั้นตอนจะอยู่หลังความเฉพาะเจาะจง แต่อยู่ก่อนลําดับการแสดง

การแสดงภาพลำดับชั้น CSS

ตามข้อกำหนด

เมื่อเปรียบเทียบการประกาศที่ปรากฏในกฎสไตล์ที่มีรูทการกําหนดขอบเขตต่างกัน การประกาศที่มีการข้ามองค์ประกอบรุ่นหรือองค์ประกอบพี่น้องน้อยที่สุดระหว่างรูทการกําหนดขอบเขตกับเรื่องของกฎสไตล์ที่มีขอบเขตจะเป็นผู้ชนะ

ขั้นตอนใหม่นี้จะมีประโยชน์เมื่อคุณฝังคอมโพเนนต์หลายรูปแบบ มาดูตัวอย่างนี้ซึ่งยังไม่ได้ใช้ @scope

<style>
    .light { background: #ccc; }
    .dark  { background: #333; }
    .light a { color: black; }
    .dark a { color: white; }
</style>
<div class="light">
    <p><a href="#">What color am I?</a></p>
    <div class="dark">
        <p><a href="#">What about me?</a></p>
        <div class="light">
            <p><a href="#">Am I the same as the first?</a></p>
        </div>
    </div>
</div>

เมื่อดูมาร์กอัปส่วนนั้น ลิงก์ที่ 3 จะเป็น white แทนที่จะเป็น black แม้ว่าจะเป็นลิงก์ย่อยของ div ที่มีการใช้คลาส .light ก็ตาม สาเหตุคือเกณฑ์ลําดับการแสดงผลที่แคสเคดใช้ที่นี่เพื่อระบุผู้ชนะ เนื่องจากระบบเห็นว่า .dark a ได้รับการประกาศเป็นอันดับสุดท้าย จึงจะชนะตามกฎ .light a

เกณฑ์การกําหนดขอบเขตความใกล้เคียงช่วยแก้ปัญหานี้ได้

@scope (.light) {
    :scope { background: #ccc; }
    a { color: black;}
}

@scope (.dark) {
    :scope { background: #333; }
    a { color: white; }
}

เนื่องจากตัวเลือก a ที่มีขอบเขตทั้ง 2 รายการมีความเฉพาะเจาะจงเหมือนกัน เกณฑ์ความใกล้เคียงของขอบเขตจึงเริ่มทำงาน โดยจะพิจารณาทั้ง 2 ตัวเลือกตามระดับความใกล้กับรูทการกําหนดขอบเขต สําหรับองค์ประกอบ a ลำดับที่ 3 นั้น องค์ประกอบดังกล่าวจะอยู่ที่ระดับ 1 ฮ็อปจากรูทการกําหนดขอบเขต .light แต่อยู่ที่ระดับ 2 ฮ็อปจากรูทการกําหนดขอบเขต .dark ดังนั้น ตัวเลือก a ใน .light จะเป็นผู้ชนะ

หมายเหตุปิดท้าย: การแยกตัวเลือก ไม่ใช่การแยกสไตล์

สิ่งที่ควรทราบอย่างหนึ่งคือ @scope จะจํากัดการเข้าถึงของตัวเลือก แต่ไม่แยกสไตล์ พร็อพเพอร์ตี้ที่รับค่าไปยังพร็อพเพอร์ตี้ย่อยจะยังคงรับค่าต่อไป นอกเหนือจากขีดจำกัดล่างของ @scope พร็อพเพอร์ตี้หนึ่งๆ ดังกล่าวคือพร็อพเพอร์ตี้ color เมื่อประกาศ color ภายในขอบเขตโดนัท color จะยังคงรับค่าจากองค์ประกอบย่อยภายในรูของโดนัท

@scope (.card) to (.card__content) {
  :scope {
    color: hotpink;
  }
}

ในตัวอย่างด้านบน องค์ประกอบ .card__content และองค์ประกอบย่อยมีสี hotpink เนื่องจากรับค่ามาจาก .card

(ภาพปกโดย rustam burkhanov ใน Unsplash)