কম্পোনেন্টে প্রপস পাস করা
React কম্পোনেন্টগুলো props ব্যবহার করে একে অপরের সাথে যোগাযোগ করে। প্রতিটি parent কম্পোনেন্ট তার child কম্পোনেন্টগুলোকে props দিয়ে কিছু তথ্য পাঠাতে পারে। Props HTML attributes-এর মত মনে হতে পারে, তবে আপনি এর মাধ্যমে object, array এবং function সহ যেকোনো JavaScript মান পাঠাতে পারেন।
যা যা আপনি শিখবেন
- কীভাবে একটি কম্পোনেন্টে props পাঠাতে হয়
 - কীভাবে একটি কম্পোনেন্ট থেকে props পড়তে হয়
 - কীভাবে props-এর জন্য default মান নির্দিষ্ট করতে হয়
 - কীভাবে একটি কম্পোনেন্টে কিছু JSX পাঠাতে হয়
 - কীভাবে সময়ের সাথে সাথে props পরিবর্তিত হয়
 
সুপরিচিত props
প্রপস হলো সেই তথ্য যা আপনি একটি JSX ট্যাগে পাঠান। উদাহরণস্বরূপ, className, src, alt, width, এবং height হল কিছু প্রপস, যা আপনি একটি <img> ট্যাগে পাঠাতে পারেন।
function Avatar() { return ( <img className="avatar" src="https://i.imgur.com/1bX5QH6.jpg" alt="Lin Lanying" width={100} height={100} /> ); } export default function Profile() { return ( <Avatar /> ); }
আপনি যে প্রপসগুলো <img> ট্যাগে পাঠাতে পারেন, সেগুলো পূর্বনির্ধারিত (ReactDOM HTML স্ট্যান্ডার্ড-এর সাথে সামঞ্জস্যপূর্ণ)। তবে আপনি নিজের কম্পোনেন্টগুলো, যেমন <Avatar>,-এ যেকোনো প্রপস পাঠাতে পারেন তাদের কাস্টমাইজ করার জন্য। এটি কীভাবে করবেন, দেখুন!
একটি কম্পোনেন্টে প্রপস পাঠানো
এই কোডে, Profile কম্পোনেন্টটি এর চাইল্ড কম্পোনেন্ট Avatar-এ কোনো প্রপস পাঠাচ্ছে না:
export default function Profile() {
  return (
    <Avatar />
  );
}আপনি দুটি ধাপে Avatar-এ কিছু props দিতে পারেন।
ধাপ ১: child কম্পোনেন্টে props পাঠানো
প্রথমে, কিছু প্রপস Avatar-এ পাঠান। উদাহরণস্বরূপ, দুটি প্রপস পাঠানো যাক: person (একটি অবজেক্ট), এবং size (একটি সংখ্যা):
export default function Profile() {
  return (
    <Avatar
      person={{ name: 'Lin Lanying', imageId: '1bX5QH6' }}
      size={100}
    />
  );
}এখন আপনি এই props Avatar কম্পোনেন্টের ভিতরে পড়তে পারেন।
ধাপ ২: চাইল্ড কম্পোনেন্টের ভেতরে প্রপস পড়ুন
আপনি function Avatar-এর ঠিক পরে ({ এবং }) এর ভিতরে props-এর নামগুলো কমা দিয়ে আলাদা করে লিখে এই props পড়তে পারেন। এভাবে, আপনি সেগুলোকে ভেরিয়েবলের মত ব্যবহার করতে পারেন।
function Avatar({ person, size }) {
  // person and size are available here
}person এবং size props ব্যবহার করে Avatar-এ কিছু logic যোগ করুন, আর আপনার কাজ শেষ।
এখন আপনি বিভিন্ন props-এর সাথে Avatar-কে বিভিন্ন উপায়ে রেন্ডার করতে পারেন। মান পরিবর্তন করে দেখুন!
import { getImageUrl } from './utils.js'; function Avatar({ person, size }) { return ( <img className="avatar" src={getImageUrl(person)} alt={person.name} width={size} height={size} /> ); } export default function Profile() { return ( <div> <Avatar size={100} person={{ name: 'Katsuko Saruhashi', imageId: 'YfeOqp2' }} /> <Avatar size={80} person={{ name: 'Aklilu Lemma', imageId: 'OKS67lh' }} /> <Avatar size={50} person={{ name: 'Lin Lanying', imageId: '1bX5QH6' }} /> </div> ); }
প্রপস আপনাকে প্যারেন্ট এবং চাইল্ড কম্পোনেন্টগুলোকে স্বাধীনভাবে ভাবতে সাহায্য করে। উদাহরণস্বরূপ, আপনি Profile-এর মধ্যে person বা size প্রপস পরিবর্তন করতে পারেন, Avatar কীভাবে সেগুলো ব্যবহার করে তা নিয়ে চিন্তা না করেই। একইভাবে, আপনি Avatar-এ প্রপসগুলো কীভাবে ব্যবহৃত হচ্ছে তা পরিবর্তন করতে পারেন, Profile-এর দিকে না তাকিয়েই।
আপনি প্রপসকে “নব” বা নিয়ন্ত্রণ কাঁটা হিসেবে ভাবতে পারেন যা আপনি সামঞ্জস্য করতে পারেন। এগুলো আসলে ফাংশনের আর্গুমেন্টের মতো কাজ করে—আসলে, প্রপস হচ্ছে আপনার কম্পোনেন্টের একমাত্র আর্গুমেন্ট! React কম্পোনেন্ট ফাংশনগুলো একটি মাত্র আর্গুমেন্ট, অর্থাৎ একটি প্রপস অবজেক্ট গ্রহণ করে।
function Avatar(props) {
  let person = props.person;
  let size = props.size;
  // ...
}সাধারণত আপনাকে সম্পূর্ণ props object প্রয়োজন হয় না, তাই আপনি এটি আলাদা props এ ভেঙে ফেলেন।
প্রপের জন্য একটি ডিফল্ট মান নির্ধারণ করা
যদি আপনি কোনো প্রপের জন্য ডিফল্ট মান নির্ধারণ করতে চান, যাতে কোনো মান নির্দিষ্ট না থাকলে তা প্রযোজ্য হয়, তাহলে ডিস্ট্রাকচারিংয়ে প্যারামিটারের পরে = এবং ডিফল্ট মানটি যোগ করতে পারেন:
function Avatar({ person, size = 100 }) {
  // ...
}এখন, যদি <Avatar person={...} /> কোনো size প্রপ ছাড়া রেন্ডার করা হয়, তাহলে size সেট হবে 100।
ডিফল্ট মানটি কেবল তখনই ব্যবহৃত হবে যদি size প্রপটি অনুপস্থিত থাকে অথবা আপনি size={undefined} পাঠান। তবে, যদি আপনি size={null} বা size={0} পাঠান, তাহলে ডিফল্ট মান ব্যবহৃত হবে না।
JSX স্প্রেড syntax এর সাথে props ফরোয়ার্ড করা
কখনও কখনও, props পাঠানো খুবই পুনরাবৃত্তিমূলক হতে পারে:
function Profile({ person, size, isSepia, thickBorder }) {
  return (
    <div className="card">
      <Avatar
        person={person}
        size={size}
        isSepia={isSepia}
        thickBorder={thickBorder}
      />
    </div>
  );
}পুনরাবৃত্তিমূলক কোডে কোনো সমস্যা নেই—এটি আরও বোধগম্য হতে পারে। কিন্তু কখনও কখনও আপনি সংক্ষিপ্ততা মূল্যায়ন করতে পারেন। কিছু কম্পোনেন্ট তাদের সমস্ত props child কম্পোনেন্টে ফরোয়ার্ড করে, যেমন Profile Avatar-এর সাথে করে। যেহেতু তারা কোনো props সরাসরি ব্যবহার করে না, তাই একটি সংক্ষিপ্ত “স্প্রেড” syntax ব্যবহার করা যেতে পারে:
function Profile(props) {
  return (
    <div className="card">
      <Avatar {...props} />
    </div>
  );
}এটি Profile এর সমস্ত props Avatar এ ফরোয়ার্ড করে কোনো নাম তালিকা ছাড়াই।
স্প্রেড syntax নিয়ন্ত্রণের সাথে ব্যবহার করুন। যদি আপনি এটি প্রতিটি কম্পোনেন্টে ব্যবহার করেন, তাহলে কিছু সমস্যা রয়েছে। সাধারণত, এটি নির্দেশ করে যে আপনাকে আপনার কম্পোনেন্টগুলো ভাগ করা উচিত এবং children JSX হিসেবে পাঠানো উচিত।
children হিসেবে JSX পাঠানো
বিল্ট-ইন ব্রাউজার ট্যাগ নেস্ট করা সাধারণ:
<div>
  <img />
</div>আপনার নিজস্ব কম্পোনেন্টগুলো একইভাবে নেস্ট করতে পারেন:
<Card>
  <Avatar />
</Card>যখন আপনি একটি JSX ট্যাগের ভিতরে কন্টেন্ট নেস্ট করেন, parent কম্পোনেন্ট সেই কন্টেন্টকে একটি children prop-এ গ্রহণ করবে। উদাহরণস্বরূপ, নিচের Card কম্পোনেন্ট একটি children prop গ্রহণ করবে যেটি <Avatar /> সেট করা আছে এবং এটি একটি wrapper div এ রেন্ডার করবে:
import Avatar from './Avatar.js'; function Card({ children }) { return ( <div className="card"> {children} </div> ); } export default function Profile() { return ( <Card> <Avatar size={100} person={{ name: 'Katsuko Saruhashi', imageId: 'YfeOqp2' }} /> </Card> ); }
Card-এর ভিতরে <Avatar> পরিবর্তন করে কিছু টেক্সট যোগানোর চেষ্টা করুন এবং দেখুন কীভাবে Card যেকোনো নেস্টেড কন্টেন্টকে র্যাপ করতে পারে। এটি জানে না যে এর ভিতরে কী রেন্ডার করা হচ্ছে। আপনি এই নমনীয় pattern অনেক জায়গায় দেখতে পাবেন।
একটি children prop সহ কম্পোনেন্টকে এমন একটি “ছিদ্র” হিসেবে ভাবতে পারেন যা parent কম্পোনেন্টগুলো যেকোনো JSX দিয়ে “পূর্ণ” করতে পারে। আপনি ভিজ্যুয়াল wrapper-দের জন্য children prop প্রায়ই ব্যবহার করবেন: প্যানেল, গ্রিড ইত্যাদি।

Illustrated by Rachel Lee Nabors
How props change over time
নিচের Clock কম্পোনেন্টটি parent কম্পোনেন্ট থেকে দুটি props গ্রহণ করে: color এবং time। (parent কম্পোনেন্টের কোডটি বাদ দেওয়া হয়েছে কারণ এটি state ব্যবহার করে, যা আমরা এখনো আলোচনা করছি না।)
নীচের সিলেক্ট বক্সে রং পরিবর্তন করার চেষ্টা করুন:
export default function Clock({ color, time }) { return ( <h1 style={{ color: color }}> {time} </h1> ); }
এই উদাহরণটি দেখায় যে একটি কম্পোনেন্ট সময়ের সাথে সাথে বিভিন্ন প্রপস পেতে পারে। প্রপস সর্বদা স্থির নয়! এখানে, time প্রপ প্রতি সেকেন্ডে পরিবর্তিত হয়, এবং color প্রপটি যখন আপনি অন্য রঙ নির্বাচন করেন তখন পরিবর্তিত হয়। প্রপস যেকোনো সময়ে একটি কম্পোনেন্টের ডেটা প্রতিফলিত করে, শুধুমাত্র শুরুতে নয়।
তবে, প্রপস অপরিবর্তনীয়—এটি একটি কম্পিউটার বিজ্ঞান সম্পর্কিত শব্দ যার অর্থ “অপরিবর্তনীয়”। যখন একটি কম্পোনেন্ট তার প্রপস পরিবর্তন করতে চায় (যেমন, ব্যবহারকারীর একটি ক্রিয়ার প্রতিক্রিয়া হিসেবে বা নতুন ডেটার জন্য), তখন এটি তার প্যারেন্ট কম্পোনেন্টকে বিভিন্ন প্রপস—একটি নতুন অবজেক্ট পাঠানোর জন্য “বিনীত” করবে! তারপর তার পুরানো প্রপস বাদ দেওয়া হবে, এবং অবশেষে জাভাস্ক্রিপ্ট ইঞ্জিন তাদের দ্বারা নেওয়া মেমরি পুনরুদ্ধার করবে।
“প্রপস পরিবর্তন করার চেষ্টা করবেন না”। যখন আপনাকে ব্যবহারকারীর ইনপুটের প্রতিক্রিয়া জানাতে হবে (যেমন নির্বাচিত রঙ পরিবর্তন করা), তখন আপনাকে “স্টেট সেট” করতে হবে, যা আপনি স্টেট: একটি কম্পোনেন্টের মেমরি থেকে শিখতে পারেন।
পুনরালোচনা
- প্রপস পাঠাতে, সেগুলো JSX-এ যুক্ত করুন, যেমন আপনি HTML অ্যাট্রিবিউটগুলোর সাথে করেন।
 - প্রপস পড়তে, 
function Avatar({ person, size })ডিস্ট্রাকচারিং সিনট্যাক্স ব্যবহার করুন। - আপনি একটি ডিফল্ট মান নির্ধারণ করতে পারেন যেমন 
size = 100, যা অনুপস্থিত এবংundefinedপ্রপসের জন্য ব্যবহৃত হয়। - আপনি সমস্ত প্রপসকে 
<Avatar {...props} />JSX স্প্রেড সিনট্যাক্সের মাধ্যমে ফরওয়ার্ড করতে পারেন, তবে এটিকে অতিরিক্ত ব্যবহার করবেন না! - নেস্টেড JSX যেমন 
<Card><Avatar /></Card>Cardকম্পোনেন্টেরchildrenপ্রপ হিসেবে উপস্থিত হবে। - প্রপস হল সময়ের একটি রিড-অনলি স্ন্যাপশট: প্রতিটি রেন্ডারে প্রপসের একটি নতুন সংস্করণ পাওয়া যায়।
 - আপনি প্রপস পরিবর্তন করতে পারবেন না। যখন আপনাকে ইন্টারঅ্যাকটিভিটি প্রয়োজন, তখন আপনাকে স্টেট সেট করতে হবে।
 
চ্যালেঞ্জ 1 / 3: একটি কম্পোনেন্ট বের করুন 
এই Gallery কম্পোনেন্টে দুটি প্রফাইলের জন্য কিছু খুব অনুরূপ মার্কআপ রয়েছে। ডুপ্লিকেশন কমানোর জন্য একটি Profile কম্পোনেন্ট বের করুন। আপনাকে এটি কী প্রপস পাঠাতে হবে তা নির্বাচন করতে হবে।
import { getImageUrl } from './utils.js'; export default function Gallery() { return ( <div> <h1>Notable Scientists</h1> <section className="profile"> <h2>Maria Skłodowska-Curie</h2> <img className="avatar" src={getImageUrl('szV5sdG')} alt="Maria Skłodowska-Curie" width={70} height={70} /> <ul> <li> <b>Profession: </b> physicist and chemist </li> <li> <b>Awards: 4 </b> (Nobel Prize in Physics, Nobel Prize in Chemistry, Davy Medal, Matteucci Medal) </li> <li> <b>Discovered: </b> polonium (chemical element) </li> </ul> </section> <section className="profile"> <h2>Katsuko Saruhashi</h2> <img className="avatar" src={getImageUrl('YfeOqp2')} alt="Katsuko Saruhashi" width={70} height={70} /> <ul> <li> <b>Profession: </b> geochemist </li> <li> <b>Awards: 2 </b> (Miyake Prize for geochemistry, Tanaka Prize) </li> <li> <b>Discovered: </b> a method for measuring carbon dioxide in seawater </li> </ul> </section> </div> ); }