अपने डीओएम के सीमित सबट्री में एलिमेंट चुनने के लिए, @scope का इस्तेमाल करने का तरीका जानें.
सीएसएस सिलेक्टर लिखने का बेहतरीन तरीका
सिलेक्टर लिखते समय, आपको दो विकल्पों में से किसी एक को चुनना पड़ सकता है. एक तरफ़, आपको यह तय करना होता है कि आपको कौनसे एलिमेंट चुनने हैं. दूसरी ओर, आपको अपने सेलेक्टर को आसानी से बदलने की सुविधा चाहिए. साथ ही, यह ज़रूरी है कि वे डीओएम स्ट्रक्चर से ज़्यादा जुड़े न हों.
उदाहरण के लिए, अगर आपको “कार्ड कॉम्पोनेंट के कॉन्टेंट एरिया में हीरो इमेज” चुननी है, तो शायद आपको .card > .content > img.hero
जैसा सिलेक्टर नहीं लिखना होगा. यह एक खास एलिमेंट का सिलेक्शन है.
- इस सिलेक्टर में
(0,3,1)
की खूबी बहुत ज़्यादा है. इसकी वजह से आपका कोड बढ़ने के साथ-साथ इसे बदलना मुश्किल हो जाता है. - डायरेक्ट चाइल्ड कॉम्बिनेटर पर भरोसा करके, यह डीओएम स्ट्रक्चर से कसकर जुड़ा होता है. अगर मार्कअप कभी बदलता है, तो आपको अपनी सीएसएस भी बदलनी होगी.
हालांकि, आपको उस एलिमेंट के लिए सिलेक्टर के तौर पर सिर्फ़ img
नहीं लिखना है, क्योंकि इससे आपके पेज पर मौजूद सभी इमेज एलिमेंट चुन लिए जाएंगे.
इस स्थिति में सही संतुलन बनाना काफ़ी चुनौती भरा होता है. पिछले कुछ सालों में, कुछ डेवलपर ने इस तरह की स्थितियों में आपकी मदद करने के लिए, समाधान और समाधान तैयार किए हैं. उदाहरण के लिए:
- BEM जैसे तरीके से यह तय किया जाता है कि आपको उस एलिमेंट को
card__img card__img--hero
की क्लास दें, ताकि आपने जो चुना है उसमें खास जानकारी शामिल न हो. - स्कोप वाली सीएसएस या स्टाइल वाले कॉम्पोनेंट जैसे JavaScript-आधारित समाधान, आपके सभी सिलेक्टर को फिर से लिखते हैं. इसके लिए, वे आपके सिलेक्टर में
sc-596d7e0e-4
जैसी स्ट्रिंग जोड़ते हैं. इससे, वे आपके पेज के दूसरी तरफ़ मौजूद एलिमेंट को टारगेट करने से बचते हैं. - कुछ लाइब्रेरी में सेलेक्टर की सुविधा पूरी तरह से बंद होती है. साथ ही, आपको स्टाइल ट्रिगर को सीधे मार्कअप में डालना पड़ता है.
लेकिन अगर आपको इनमें से किसी की भी ज़रूरत नहीं है, तो क्या होगा? क्या होगा, अगर सीएसएस आपको यह तय करने का विकल्प दे कि आपको कौनसे एलिमेंट चुनने हैं. इसके लिए, आपको ज़्यादा सटीक सिलेक्टर लिखने या ऐसे सिलेक्टर लिखने की ज़रूरत नहीं होगी जो आपके डीओएम से ज़्यादा जुड़े हों. ऐसे में, @scope
का इस्तेमाल करके, सिर्फ़ अपने DOM के सबट्री में मौजूद एलिमेंट चुने जा सकते हैं.
पेश है @scope
@scope
का इस्तेमाल करके, अपने सिलेक्टर की पहुंच को सीमित किया जा सकता है. ऐसा करने के लिए, स्कोपिंग रूट सेट करें. इससे, उस सबट्री की ऊपरी सीमा तय होती है जिसे आपको टारगेट करना है. स्कोप वाले रूट सेट में मौजूद स्टाइल नियम, सिर्फ़ DOM के उस सीमित सबट्री से चुने जा सकते हैं जिसे स्कोप वाले स्टाइल नियम कहा जाता है.
उदाहरण के लिए, .card
कॉम्पोनेंट में सिर्फ़ <img>
एलिमेंट को टारगेट करने के लिए, .card
को @scope
at-rule के स्कोपिंग रूट के तौर पर सेट किया जाता है.
@scope (.card) {
img {
border-color: green;
}
}
स्कोप वाले स्टाइल नियम img { … }
से, सिर्फ़ वे <img>
एलिमेंट चुने जा सकते हैं जो मैच किए गए .card
एलिमेंट के स्कोप में हों.
कार्ड के कॉन्टेंट एरिया (.card__content
) में मौजूद <img>
एलिमेंट के चुने जाने से रोकने के लिए, 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
से पहले, इसके बारे में साफ़ तौर पर बताया जा सकता है. इसके अलावा, सीएसएस नेस्टिंग से &
सिलेक्टर को पहले से जोड़ा जा सकता है.
@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
@scope
, सीएसएस कैस्केड में एक नई शर्त भी जोड़ता है: स्कोपिंग प्रॉक्सिमिटी. यह चरण, खास जानकारी के बाद आता है, लेकिन दिखने के क्रम से पहले.
अलग-अलग स्कोपिंग रूट वाले स्टाइल नियमों में दिखने वाले एलान की तुलना करते समय, स्कोपिंग रूट और स्कोप वाले स्टाइल नियम के विषय के बीच, सबसे कम जनरेशनल या सिबलिंग-एलिमेंट वाली होप वाला एलान जीतता है.
यह नया चरण, किसी कॉम्पोनेंट के कई वैरिएंट को नेस्ट करते समय काम आता है. इस उदाहरण में, @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>
मार्कअप का यह छोटा सा हिस्सा देखते समय, तीसरा लिंक black
के बजाय white
होगा. भले ही, यह div
का चाइल्ड एलिमेंट है और उस पर क्लास .light
लागू है. ऐसा, विज्ञापन दिखने के क्रम की शर्त की वजह से होता है. विजेता तय करने के लिए, कैस्केड यहां इसका इस्तेमाल करता है. यह देखता है कि .dark a
को आखिर में घोषित किया गया था, इसलिए यह .light a
नियम के तहत जीत जाएगा
स्कोपिंग के लिए, जगह की निकटता से जुड़ी शर्त की मदद से, अब यह समस्या हल हो गई है:
@scope (.light) {
:scope { background: #ccc; }
a { color: black;}
}
@scope (.dark) {
:scope { background: #333; }
a { color: white; }
}
स्कोप वाले दोनों a
सिलेक्टर की खास बात एक जैसी होने की वजह से, स्कोपिंग प्रॉक्सिमिटी क्राइटेरियम काम करना शुरू कर देता है. यह स्कोपिंग रूट के करीब होने के हिसाब से, दोनों सिलेक्टर का वज़न तय करता है. तीसरे a
एलिमेंट के लिए, .light
स्कोपिंग रूट तक सिर्फ़ एक हॉप है, लेकिन .dark
रूट तक दो हॉप हैं. इसलिए, .light
में मौजूद a
सिलेक्टर जीतेगा.
आखिरी बात: स्टाइल आइसोलेशन नहीं, बल्कि सिलेक्टर आइसोलेशन
एक अहम बात यह है कि @scope
, सिलेक्टर की पहुंच को सीमित करता है. यह स्टाइल अलग करने की सुविधा नहीं देता. @scope
की निचली सीमा के पार, ऐसी प्रॉपर्टी जो बच्चों के लिए इनहेरिट की जाती हैं वे अब भी इनहेरिट की जाएंगी. ऐसी ही एक प्रॉपर्टी color
है. अगर किसी डोनट स्कोप में एक एलिमेंट का एलान किया जाता है, तो color
अब भी डोनट के होल में मौजूद चाइल्ड एलिमेंट को इनहेरिट करेगा.
@scope (.card) to (.card__content) {
:scope {
color: hotpink;
}
}
ऊपर दिए गए उदाहरण में, .card__content
एलिमेंट और उसके चाइल्ड एलिमेंट का रंग hotpink
है, क्योंकि वे .card
से वैल्यू इनहेरिट करते हैं.
(कवर फ़ोटो, rustam burkhanov की है. इसे Unsplash से लिया गया है)