डेटा को कॉन्टेक्स्ट का उपयोग करके गहराई तक भेजना
आमतौर पर, आप जानकारी को एक पैरेंट कौम्पोनॅन्ट से एक चाइल्ड कौम्पोनॅन्ट तक प्रॉप्स के माध्यम से भेजते हैं। लेकिन प्रॉप्स को भेजना वाचाल और असुविधाजनक बन सकता है यदि आपको उन्हें बीच में कई कॉम्पोनेंट्स से भेजना पड़ता है, या यदि आपके ऐप में कई कॉम्पोनेंट्स को समान जानकारी की आवश्यकता है। कॉन्टेक्स्ट पैरेंट कौम्पोनॅन्ट को उसके नीचे के पेड़ में किसी भी कॉन्टेक्स्ट के लिए कुछ जानकारी उपलब्ध कराने देता है - चाहे वह कितना भी गहरा हो - बिना उसे प्रॉप्स के द्वारा स्पष्ट रूप से पास किये।
You will learn
- “प्रोप ड्रिलिंग” क्या है
- कॉन्टेक्स्ट के द्वारा प्रोप का बार-बार करना कैसे बदलें
- कॉन्टेक्स्ट के लिए सामान्य उपयोग
- कॉन्टेक्स्ट के सामान्य विकल्प
प्रॉप्स को पास करने में समस्या
प्रॉप्स को पास करना अपने UI ट्री के माध्यम से स्पष्ट रूप से डेटा को उन कॉम्पोनेंट्स को पाइप करने का एक शानदार तरीका है जो इसका उपयोग करते हैं।
लेकिन प्रॉप्स को पास करना वाचाल और असुविधाजनक बन सकता है जब आपको पेड़ के माध्यम से कुछ प्रोप को गहराई तक पास करने की आवश्यकता होती है, या यदि कई कॉम्पोनेंट्स को एक ही प्रोप की आवश्यकता होती है। निकटतम एंसेस्टर को उन कॉम्पोनेंट्स से दूर किया जा सकता है जिन्हें डेटा की आवश्यकता है, और उतनी ऊँची स्टेट को उठाना “प्रोप ड्रिलिंग” नामक स्थिति को जन्म दे सकता है।
स्टेट को ऊपर उठाना


प्रोप ड्रिलिंग


क्या यह बहुत अच्छा नहीं होगा यदि पेड़ में कॉम्पोनेंट्स ko बिना प्रॉप्स के पास किये डेटा “टेलीपोर्ट” करने का एक तरीका होता जिनको उसकी जरुरत है? React के कॉन्टेक्स्ट फ़ीचर के साथ, यह संभव है!
कॉन्टेक्स्ट: प्रॉप्स को भेजने का एक विकल्प
कॉन्टेक्स्ट एक पैरेंट कौम्पोनॅन्ट को उसके नीचे पूरे पेड़ को डेटा प्रदान करने देता है। कॉन्टेक्स्ट के कई उपयोग हैं। यहाँ एक उदाहरण है। इस Heading कौम्पोनॅन्ट पर विचार करें जो इसके आकार के लिए एक level स्वीकार करता है:
import Heading from './Heading.js'; import Section from './Section.js'; export default function Page() { return ( <Section> <Heading level={1}>Title</Heading> <Heading level={2}>Heading</Heading> <Heading level={3}>Sub-heading</Heading> <Heading level={4}>Sub-sub-heading</Heading> <Heading level={5}>Sub-sub-sub-heading</Heading> <Heading level={6}>Sub-sub-sub-sub-heading</Heading> </Section> ); }
मान लीजिए कि आप एक ही Section के भीतर कई शीर्षकों का हमेशा एक ही आकार चाहते हैं:
import Heading from './Heading.js'; import Section from './Section.js'; export default function Page() { return ( <Section> <Heading level={1}>Title</Heading> <Section> <Heading level={2}>Heading</Heading> <Heading level={2}>Heading</Heading> <Heading level={2}>Heading</Heading> <Section> <Heading level={3}>Sub-heading</Heading> <Heading level={3}>Sub-heading</Heading> <Heading level={3}>Sub-heading</Heading> <Section> <Heading level={4}>Sub-sub-heading</Heading> <Heading level={4}>Sub-sub-heading</Heading> <Heading level={4}>Sub-sub-heading</Heading> </Section> </Section> </Section> </Section> ); }
वर्तमान में, आप level प्रोप को अलग से प्रत्येक <Heading> में पास करते हैं:
<Section>
<Heading level={3}>About</Heading>
<Heading level={3}>Photos</Heading>
<Heading level={3}>Videos</Heading>
</Section>यह अच्छा होगा यदि आप level प्रोप को इसके बजाय <Section> कौम्पोनॅन्ट में पास कर सकते और इसे <Heading> से हटा सकते। इस प्रकार आप यह लागू कर सकते हैं कि एक ही सैक्शन में सभी शीर्षकों का आकार समान है:
<Section level={3}>
<Heading>About</Heading>
<Heading>Photos</Heading>
<Heading>Videos</Heading>
</Section>लेकिन <Heading> कौम्पोनॅन्ट अपने निकटतम <Section> के स्तर को कैसे जान सकता है? इसके लिए एक चाइल्ड के लिए पेड़ में कहीं ऊपर से डेटा के लिए “पूछने” के लिए किसी तरह की आवश्यकता होगी।
आप इसे अकेले प्रॉप्स के साथ नहीं कर सकते। यह वह जगह है जहां कॉन्टेक्स्ट काम आता है। आप इसे तीन स्टेप में करेंगे:
- कॉन्टेक्स्ट को बनाएं।. (आप इसे
LevelContextबुला सकते हैं, चूंकि यह शीर्षक स्तर के लिए है।) - उस कौम्पोनॅन्ट से उस कॉन्टेक्स्ट का उपयोग करें जिसे डेटा की आवश्यकता है। (
HeadingLevelContextका प्रयोग करेगा।) - डेटा को निर्दिष्ट करने वाले कौम्पोनॅन्ट से उस कॉन्टेक्स्ट को प्रदान करें। (
SectionLevelContextको प्रदान करेगा।)
कॉन्टेक्स्ट एक पैरेंट—यहां तक कि बहुत दूर वाला भी!—उसके अंदर पूरे पेड़ को कुछ डेटा प्रदान करने देता है।
करीबी चिल्ड्रन में कॉन्टेक्स्ट का उपयोग करना


दूर के चिल्ड्रन में कॉन्टेक्स्ट का उपयोग करना


स्टेप 1: कॉन्टेक्स्ट बनाएं
सबसे पहले, आपको कॉन्टेक्स्ट बनाने की आवश्यकता है। आपको इसे एक फ़ाइल से निर्यात करने की आवश्यकता होगी ताकि आपके कॉम्पोनेंट्स इसका उपयोग कर सकें:
import { createContext } from 'react'; export const LevelContext = createContext(1);
CreateContext का एकमात्र आर्ग्युमेंट डिफ़ॉल्ट वैल्यू है। यहाँ, 1 सबसे बड़े हेडिंग स्तर को संदर्भित करता है, लेकिन आप किसी भी तरह के मूल्य (यहां तक कि एक ऑब्जेक्ट) को पास कर सकते हैं। आप अगले स्टेप में डिफ़ॉल्ट वैल्यू का महत्व देखेंगे।
स्टेप 2: कॉन्टेक्स्ट का उपयोग करें
React के useContext hook को और अपने कॉन्टेक्स्ट को आयात करें:
import { useContext } from 'react';
import { LevelContext } from './LevelContext.js';वर्तमान में, Heading कौम्पोनॅन्ट प्रॉप्स से level प्राप्त करता है:
export default function Heading({ level, children }) {
// ...
}इसके बजाय, level प्रोप को हटा दें और आपके द्वारा इम्पोर्ट किए गए कॉन्टेक्स्ट LevelContext से वैल्यू रीड करें:
export default function Heading({ children }) {
const level = useContext(LevelContext);
// ...
}useContext एक Hook है। useState और useReducer की तरह ही, आप एक React कौम्पोनॅन्ट के अंदर एक Hook को तुरंत ही कॉल कर सकते हैं (loops या conditions के अंदर नहीं)। useContext React को बताता है कि Heading कौम्पोनॅन्ट LevelContext को पढ़ना चाहता है।
अब जब Heading कौम्पोनॅन्ट में level प्रोप नहीं है, तो आपको अपने JSX में Heading को level प्रोप को इस प्रकार से अब पास करने की आवश्यकता नहीं है:
<Section>
<Heading level={4}>Sub-sub-heading</Heading>
<Heading level={4}>Sub-sub-heading</Heading>
<Heading level={4}>Sub-sub-heading</Heading>
</Section>JSX को अपडेट करें ताकि यह इसके बजाय अब Section हो जो इसे प्राप्त करे:
<Section level={4}>
<Heading>Sub-sub-heading</Heading>
<Heading>Sub-sub-heading</Heading>
<Heading>Sub-sub-heading</Heading>
</Section>एक अनुस्मारक के रूप में, यह मार्कअप है जिसे आप सही काम करने की कोशिश करा रहे थे:
import Heading from './Heading.js'; import Section from './Section.js'; export default function Page() { return ( <Section level={1}> <Heading>Title</Heading> <Section level={2}> <Heading>Heading</Heading> <Heading>Heading</Heading> <Heading>Heading</Heading> <Section level={3}> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Section level={4}> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> </Section> </Section> </Section> </Section> ); }
ध्यान दें कि यह उदाहरण अब भी काफी काम नहीं करता है! सभी शीर्षकों का एक ही आकार है क्योंकि भले ही आप कॉन्टेक्स्ट का उपयोग कर रहे हैं, आपने इसे अभी तक प्रदान नहीं किया है। react को पता नहीं है कि इसे कहां से प्राप्त करना है!
यदि आप कॉन्टेक्स्ट प्रदान नहीं करते हैं, तो React पिछले स्टेप में आपके द्वारा निर्दिष्ट डिफ़ॉल्ट मान का उपयोग करेगा। इस उदाहरण में, आपने 1 को createContext के आर्ग्युमेंट के रूप में निर्दिष्ट किया था, इसलिए useContext(LevelContext) 1 रिटर्न करता है, जिससे वे सभी शीर्षक <h1> पर सेट हो गए हैं। आइए इस समस्या को प्रत्येक Section अपना कॉन्टेक्स्ट प्रदान कराने के द्वारा ठीक करें ।
स्टेप 3: कॉन्टेक्स्ट प्रदान करें
अभी Section कौम्पोनॅन्ट अपने चिल्ड्रन को रेंडर करता है:
export default function Section({ children }) {
return (
<section className="section">
{children}
</section>
);
}उन्हें LevelContext प्रदान करने के लिए उन्हें एक कॉन्टेक्स्ट प्रदाता के साथ लपेटें:
import { LevelContext } from './LevelContext.js';
export default function Section({ level, children }) {
return (
<section className="section">
<LevelContext value={level}>
{children}
</LevelContext>
</section>
);
}यह React को बताता है: “यदि इस <Section> के अंदर कोई भी कौम्पोनॅन्ट LevelContext के लिए पूछता है, तो उन्हें यह level दें।” कौम्पोनॅन्ट इसके ऊपर UI पेड़ में निकटतम <LevelContext> के मान का उपयोग करेगा।
import Heading from './Heading.js'; import Section from './Section.js'; export default function Page() { return ( <Section level={1}> <Heading>Title</Heading> <Section level={2}> <Heading>Heading</Heading> <Heading>Heading</Heading> <Heading>Heading</Heading> <Section level={3}> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Section level={4}> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> </Section> </Section> </Section> </Section> ); }
यह मूल कोड के समान परिणाम है, लेकिन आपको प्रत्येक Heading कौम्पोनॅन्ट को level प्रोप को पास करने की आवश्यकता नहीं पड़ी! इसके बजाय, यह अपने ऊपर निकटतम Section से पूछकर अपने शीर्षक स्तर का “पता कर लेता है”:
- आप
<Section>में एकlevelप्रोप को पास करते हैं। Sectionअपने चिल्ड्रन को<LevelContext value={level}>में लपेटता है।HeadingLevelContextके निकटतम मूल्य कोuseContext(LevelContext)का उपयोग करके ऊपर पूछता है।
एक ही कौम्पोनॅन्ट से कॉन्टेक्स्ट का उपयोग करना और प्रदान करना
वर्तमान में, आपको अभी भी प्रत्येक सेक्शन का level खुद से निर्दिष्ट करना पड़ रहा है:
export default function Page() {
return (
<Section level={1}>
...
<Section level={2}>
...
<Section level={3}>
...चूंकि कॉन्टेक्स्ट आपको ऊपर एक कौम्पोनॅन्ट से जानकारी पढ़ने देता है, प्रत्येक Section ऊपर वाले Section से level पढ़ सकता है, और स्वचालित रूप से level + 1 नीचे पास कर सकता है। यहां बताया गया है कि आप इसे कैसे कर सकते हैं:
import { useContext } from 'react';
import { LevelContext } from './LevelContext.js';
export default function Section({ children }) {
const level = useContext(LevelContext);
return (
<section className="section">
<LevelContext value={level + 1}>
{children}
</LevelContext>
</section>
);
}इस परिवर्तन के साथ, आपको level प्रोप को न ही <Section> न ही <Heading> में पास करने की आवश्यकता है।
import Heading from './Heading.js'; import Section from './Section.js'; export default function Page() { return ( <Section> <Heading>Title</Heading> <Section> <Heading>Heading</Heading> <Heading>Heading</Heading> <Heading>Heading</Heading> <Section> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Heading>Sub-heading</Heading> <Section> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> <Heading>Sub-sub-heading</Heading> </Section> </Section> </Section> </Section> ); }
अब दोनों Heading और Section LevelContext को पढ़ते है यह पता लगाने के लिए कि वे कितने “गहरे” हैं। और Section अपने चिल्ड्रन को LevelContext में लपेटता है, यह निर्दिष्ट करने के लिए कि इसके अंदर कुछ भी “गहरे” स्तर पर है।
कॉन्टेक्स्ट मध्यवर्ती कौम्पोनॅन्ट कॉम्पोनेंट्स से गुजरता है
आप उन कॉम्पोनेंट्स के बीच में कई कॉम्पोनेंट्स को सम्मिलित कर सकते हैं जो कॉन्टेक्स्ट प्रदान करता है और जो इसका उपयोग करता है। इसमें <div> जैसे अंतर्निहित कॉम्पोनेंट्स और वे कॉम्पोनेंट्स जो आप स्वयं बना सकते हैं, दोनों शामिल हैं।
इस उदाहरण में, एक ही Post कौम्पोनॅन्ट (एक धराशायी सीमा के साथ) को दो अलग -अलग nesting स्तरों पर रेंडर किया गया है। ध्यान दें कि इसके अंदर <Heading> को इसका स्तर निकटतम <Section> से स्वचालित रूप से मिल जाता है:
import Heading from './Heading.js'; import Section from './Section.js'; export default function ProfilePage() { return ( <Section> <Heading>My Profile</Heading> <Post title="Hello traveller!" body="Read about my adventures." /> <AllPosts /> </Section> ); } function AllPosts() { return ( <Section> <Heading>Posts</Heading> <RecentPosts /> </Section> ); } function RecentPosts() { return ( <Section> <Heading>Recent Posts</Heading> <Post title="Flavors of Lisbon" body="...those pastéis de nata!" /> <Post title="Buenos Aires in the rhythm of tango" body="I loved it!" /> </Section> ); } function Post({ title, body }) { return ( <Section isFancy={true}> <Heading> {title} </Heading> <p><i>{body}</i></p> </Section> ); }
आपने इसे काम करने के लिए कुछ खास नहीं किया। एक Section उसके अंदर के पेड़ के लिए कॉन्टेक्स्ट को निर्दिष्ट करता है, जिससे आप कहीं भी एक <Heading> डाल सकते हैं, और इसका सही आकार होगा। इसे ऊपर सैंडबॉक्स में आज़माएं!
कॉन्टेक्स्ट आपको ऐसे कौम्पोनॅन्ट को लिखने देता है जो “अपने परिवेश के अनुकूल बन जाते हैं” और खुद को अलग -अलग तरीके से प्रदर्शित करते हैं जो इस बात पर निर्भर करता है की वे कहाँ (या, दूसरे शब्दों में, किस कॉन्टेक्स्ट में) रेंडर हो रहे हैं।
कैसे कॉन्टेक्स्ट काम करता है आपको CSS property inheritance की याद दिला सकता है। CSS में आप एक <div> और उसके अंदर किसी भी DOM node के लिए color: blue को निर्दिष्ट कर सकते हैं। इसी तरह, React में, ऊपर से आने वाले कुछ कॉन्टेक्स्ट को ओवरराइड करने का एकमात्र तरीका चिल्ड्रन को एक अलग मूल्य के साथ एक कॉन्टेक्स्ट प्रदाता में लपेटना है।
CSS में, color और background-color जैसे विभिन्न प्रोपर्टीज़ एक दूसरे को ओवरराइड नहीं करती हैं। आप background-color को प्रभावित किए बिना सभी <div> के color को red सेट कर सकते हैं। इसी प्रकार, अलग-अलग React कॉन्टेक्स्ट एक-दूसरे को ओवरराइड नहीं करते हैं। प्रत्येक कॉन्टेक्स्ट जो आप createContext() का प्रयोग कर बनाते हैं, अन्य कॉन्टेक्स्ट से पूरी तरह से अलग होता है, और उन कॉम्पोनेंट्स को जो उस विशिष्ट कॉन्टेक्स्ट का उपयोग और प्रदान करते हैं, एक साथ बांधता है। एक कौम्पोनॅन्ट बिना किसी समस्या के कई अलग-अलग कॉन्टेक्स्ट का उपयोग या प्रदान कर सकता है।
इससे पहले कि आप कॉन्टेक्स्ट का उपयोग करें
कॉन्टेक्स्ट का उपयोग करना बहुत लुभावना है! हालांकि, इसका मतलब यह भी है कि इसे अति प्रयोग करना बहुत आसान है। सिर्फ इसलिए कि आपको कई गहरे स्तरों तक कुछ प्रॉप्स पास करने की आवश्यकता का यह मतलब नहीं है कि आपको उस जानकारी को कॉन्टेक्स्ट में रखना चाहिए।
कॉन्टेक्स्ट का उपयोग करने से पहले आपको इन कुछ विकल्पों पर विचार करना चाहिए:
- प्रॉप्स को पास करने से शुरू करें। यदि आपके कॉम्पोनेंट्स तुच्छ नहीं हैं, तो दर्जन भर कॉम्पोनेंट्स के माध्यम से दर्जन भर प्रॉप्स को पास करना असामान्य नहीं है। यह एक नारे की तरह लग सकता है, लेकिन यह बहुत स्पष्ट करता है कि कौन से कॉम्पोनेंट्स किस डेटा का उपयोग करते हैं! आपके कोड को मेन्टेन रखने वाले व्यक्ति को खुशी होगी कि आपने डेटा प्रवाह को प्रॉप्स के साथ स्पष्ट कर दिया है।
- कॉम्पोनेंट्स को एक्सट्रेक्ट करें और JSX को
चिल्ड्रनके रूप में उनमे भेजें। यदि आप मध्यवर्ती कॉम्पोनेंट्स की कई परतों के माध्यम से कुछ डेटा पास करते हैं जो उस डेटा का उपयोग नहीं करते हैं (और केवल इसे और नीचे पास करते हैं), तो इसका मतलब है कि आप कुछ कॉम्पोनेंट्स को एक्सट्रेक्ट करने के लिए भूल गए। उदाहरण के लिए, हो सकता है कि आप दृश्य कॉम्पोनेंट्स के लिएpostsजैसे डेटा प्रॉप्स पास करते हैं जो उन्हें सीधे उपयोग नहीं करते हैं, जैसे कि<Layout posts={posts} />। इसके बजाय,Layoutकोchildrenको एक प्रोप के रूप में लेने के लिए बनाएं, और<Layout><Posts posts={posts} /></Layout>को रेंडर करें। यह डेटा को निर्दिष्ट करने वाले कौम्पोनॅन्ट और डेटा की आवश्यकता वाले कौम्पोनॅन्ट के बीच परतों की संख्या को कम करता है।
यदि इनमें से कोई भी दृष्टिकोण आपके लिए अच्छा काम नहीं करता है, तो कॉन्टेक्स्ट पर विचार करें।
कॉन्टेक्स्ट के लिए उपयोग के मामले
- थीम: यदि आपका ऐप उपयोगकर्ता को अपनी उपस्थिति (जैसे डार्क मोड) को बदलने देता है, तो आप अपने ऐप के शीर्ष पर एक कॉन्टेक्स्ट प्रदाता रख सकते हैं, और उस कॉन्टेक्स्ट का उपयोग उन कॉन्टेक्स्ट में कर सकते हैं जिन्हें अपने दृश्य लुक को समायोजित करने की आवश्यकता है।
- चालू खाता: कई कॉम्पोनेंट्स को वर्तमान में logged in उपयोगकर्ता का पता करने की आवश्यकता हो सकती है। इसे कॉन्टेक्स्ट में रखना पेड़ में कहीं भी इसे पढ़ने के लिए सुविधाजनक बनाता है। कुछ ऐप्स आपको एक ही समय में कई खाते संचालित करने देते हैं (उदाहरण के लिए एक अलग उपयोगकर्ता के रूप में एक टिप्पणी छोड़ने के लिए)। उन मामलों में, एक अलग चालू खाता मूल्य के साथ एक नेस्टेड कॉन्टेक्स्ट प्रदाता में UI के एक हिस्से को लपेटना सुविधाजनक हो सकता है।
- राउटिंग: अधिकांश राउटिंग समाधान वर्तमान मार्ग को रखने के लिए आंतरिक रूप से कॉन्टेक्स्ट का उपयोग करते हैं। इस प्रकार से हर लिंक “जानता है” कि यह सक्रिय है या नहीं। यदि आप अपना खुद का राउटर बनाते हैं, तो आप इसे भी करना चाह सकते हैं।
- स्टेट को मैनेज करना: जैसे-जैसे आपका ऐप बढ़ता है, आप अपने ऐप के शीर्ष के करीब बहुत सारे स्टेट के साथ समाप्त हो सकते हैं। नीचे कई दूर के कौम्पोनॅन्ट इसे बदलना चाह सकते हैं। जटिल स्टेट को मैनेज करने के लिए कॉन्टेक्स्ट के साथ एक रेड्यूसर का उपयोग करना और और इसे बहुत अधिक परेशानी के बिना दूर के कॉम्पोनेंट्स तक पास करना सामान्य है।
कॉन्टेक्स्ट स्थैतिक मूल्यों तक सीमित नहीं है। यदि आप अगले रेंडर पर एक अलग मूल्य पास करते हैं, तो React नीचे पढ़ते हुए सभी कॉम्पोनेंट्स को अपडेट करेगा! यही कारण है कि कॉन्टेक्स्ट का उपयोग अक्सर स्टेट के साथ संयोजन में किया जाता है।
सामान्य तौर पर, यदि पेड़ के विभिन्न हिस्सों में दूर के कॉम्पोनेंट्स द्वारा कुछ जानकारी की आवश्यकता होती है, तो यह एक अच्छा संकेत है कि कॉन्टेक्स्ट आपकी मदद करेगा।
Recap
- कॉन्टेक्स्ट एक कौम्पोनॅन्ट को इसके नीचे पूरे पेड़ को कुछ जानकारी प्रदान करता है।
- कॉन्टेक्स्ट को पास करने के लिए:
- इसे बनाएं और निर्यात करें
export const MyContext = createContext(defaultValue). - इसे किसी भी चाइल्ड कौम्पोनॅन्ट में पढ़ने के लिए
useContext(MyContext)हुक में पास करें, चाहे वह कितना भी गहरा क्यों न हो। - चिल्ड्रन को पैरेंट से प्रदान करने के लिए
<MyContext value={...}>में लपेटें।
- इसे बनाएं और निर्यात करें
- कॉन्टेक्स्ट बीच में किसी भी कॉम्पोनेंट्स से गुजरता है।
- कॉन्टेक्स्ट आपको उन कॉम्पोनेंट्स को लिखने देता है जो “अपने परिवेश के अनुकूल हो जाते” हैं।
- कॉन्टेक्स्ट का उपयोग करने से पहले, प्रॉप्स पास करने या JSX को
childrenके रूप में पास करने का प्रयास करें।
Challenge 1 of 1: प्रॉप ड्रिलिंग को कॉन्टेक्स्ट से बदलें
इस उदाहरण में, चेकबॉक्स को टॉगल करने से प्रत्येक <PlaceImage> के लिए पारित imageSize प्रोप बदल जाता है। चेकबॉक्स स्टेट शीर्ष-स्तरीय App कौम्पोनॅन्ट में रखा जाता है, लेकिन प्रत्येक <PlaceImage> को इसके बारे में पता होना चाहिए।
वर्तमान में, App List में imageSize को पास करता है, जो इसे प्रत्येक Place को पास करता है, जो इसे PlaceImage को पास करता है। imageSize प्रोप को हटा दें, और इसके बजाय इसे App कौम्पोनॅन्ट से सीधे PlaceImage तक पास करें।
आप Context.js में कॉन्टेक्स्ट घोषित कर सकते हैं।
import { useState } from 'react'; import { places } from './data.js'; import { getImageUrl } from './utils.js'; export default function App() { const [isLarge, setIsLarge] = useState(false); const imageSize = isLarge ? 150 : 100; return ( <> <label> <input type="checkbox" checked={isLarge} onChange={e => { setIsLarge(e.target.checked); }} /> Use large images </label> <hr /> <List imageSize={imageSize} /> </> ) } function List({ imageSize }) { const listItems = places.map(place => <li key={place.id}> <Place place={place} imageSize={imageSize} /> </li> ); return <ul>{listItems}</ul>; } function Place({ place, imageSize }) { return ( <> <PlaceImage place={place} imageSize={imageSize} /> <p> <b>{place.name}</b> {': ' + place.description} </p> </> ); } function PlaceImage({ place, imageSize }) { return ( <img src={getImageUrl(place)} alt={place.name} width={imageSize} height={imageSize} /> ); }