القوائم و المفاتيح
These docs are old and won’t be updated. Go to react.dev for the new React docs.
These new documentation pages teach modern React and include live examples:
فلنتذكّر أولًا كيفيّة تحويل القوائم في JavaScript.
في المثال التالي سنستخدم الدالة ()map
لمضاعفة قيم مصفوفة من الأرقام اسمها numbers
، وسنُعيِّن المصفوفة الجديدة التي تُعيدها الدالة map ()
إلى المتغير doubled
ثم نعرض محتواه عبر التابع console.log()
:
const numbers = [1, 2, 3, 4, 5];
const doubled = numbers.map((number) => number * 2);console.log(doubled);
نتيجة تنفيذ هذا المثال هي [2 ، 4 ، 6 ، 8 ، 10]
.
يكون تحويل المصفوفات في React إلى قوائم من العناصر مماثلًا تقريبًا لذلك.
تصيير عدة مكونات
يمكنك بناء مجموعة من عناصر و إحوائهم في JSX بإستخدام العارضتين {}
.
في المثال التالي سنستخدم الدالة map()
للمرور على جميع عناصر مصفوفة الأرقام numbers
بحيث نُعيد عنصر <li>
لكل عنصر من هذه المصفوفة. وأخيرا نُعيِّن مصفوفة العناصر الناتجة إلى المتغير listItems
:
const numbers = [1, 2, 3, 4, 5];
const listItems = numbers.map((number) => <li>{number}</li>);
نضع الآن كامل المصفوفة listItems
بداخل عنصر <ul>
، ونُصيِّره إلى DOM:
<ul>{listItems}</ul>
تعرض هذه الشيفرة قائمة منقطة مؤلفة من الأرقام بين 1 و 5.
مكون يحتوي على قائمة بسيطة
نُصيِّر عادةً القوائم بداخل المُكوِّنات. لذلك بإمكاننا إعادة كتابة المثال السابق باستخدام مُكوِّن يقبل مصفوفة من الأرقام تُدعى numbers
ويعرض عناصرها على شكل قائمة غير مرتبة:
function NumberList(props) {
const numbers = props.numbers;
const listItems = numbers.map((number) => <li>{number}</li> ); return (
<ul>{listItems}</ul> );
}
const numbers = [1, 2, 3, 4, 5];
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(<NumberList numbers={numbers} />);
عندما تُنفَّذ هذه الشيفرة ستتلقى تحذيرًا أنّه يجب تزويد مفتاح (key) لعناصر القائمة. والمفتاح هو عبارة عن خاصيّة على شكل سلسلة نصيّة يجب إضافتها عند إنشاء قوائم من العناصر. سنتحدث لاحقا عن أهميته في موضوع القادم.
فلنُعيِّن مفتاح key
لعناصر القائمة بداخل التابع numbers.map()
وبذلك نُصلِح مشكلة المفتاح المفقود:
function NumberList(props) {
const numbers = props.numbers;
const listItems = numbers.map((number) =>
<li key={number.toString()}> {number}
</li>
);
return (
<ul>{listItems}</ul>
);
}
المفاتيح (Keys)
تُساعِد المفاتيح React على معرفة العناصر التي تغيرت، أو أُضيفت، أو أُزيلت. يجب أن تُعطَى المفاتيح للعناصر بداخل المصفوفة وذلك لإعطاء هذه العناصر هوية مستقرة:
const numbers = [1, 2, 3, 4, 5];
const listItems = numbers.map((number) =>
<li key={number.toString()}> {number}
</li>
);
أفضل طريقة لانتقاء مفتاح هي استخدام سلسلة نصيّة تُعرِّف بشكل فريد عنصر القائمة عن العناصر الأخرى، نستخدم غالبًا المُعرِّفات (IDs) من بياناتنا كمفاتيح:
const todoItems = todos.map((todo) =>
<li key={todo.id}> {todo.text}
</li>
);
عندما لا تملك مُعرِّفات مستقرة للعناصر، فبإمكانك استخدام فهرس العنصر index
كمفتاح كملاذ أخير:
const todoItems = todos.map((todo, index) =>
// افعل ذلك فقط إن لم لكن للعناصر معرّفات مستقرة <li key={index}> {todo.text}
</li>
);
لا نُفضِّل استخدام فهارس العناصر إن كان ترتيبها عُرضةً للتغيير، فقد يُؤثِّر هذا بشكل سلبي على الأداء وقد يسبب مشاكل مع حالة المُكوِّن. اطّلع على هذا المقال للحصول على شرح مُفصّل للتأثيرات السلبية لاستخدام الفهرس كمفتاح . إن اخترت عدم تعيين مفتاح لعناصر القائمة فستستخدم React الفهارس كمفاتيح بشكل افتراضي.
إن كنت ترغب بمعرفة المزيد ستجد هنا شرحا مفصلا حول أهمية المفاتيح.
استخراج المكونات ذات المفاتيح
يكون للمفاتيح معنى فقط في سياق المصفوفة الموجودة ضمنها.
فمثلًا إن استخرجت المُكوِّن ListItem
فيجب أن تحتفظ بالمفتاح ضمن العناصر <ListItem />
الموجودة في المصفوفة بدلًا من وضعه في العنصر <li>
لموجود في المُكوِّن ListItem
.
مثال: الاستخدام الخاطئ للمفتاح
function ListItem(props) {
const value = props.value;
return (
// .خطأ! فليس هنالك حاجة لتحديد المفتاح هنا <li key={value.toString()}> {value}
</li>
);
}
function NumberList(props) {
const numbers = props.numbers;
const listItems = numbers.map((number) =>
// .خطأ! فليس هنالك حاجة لتحديد المفتاح هنا <ListItem value={number} /> );
return (
<ul>
{listItems}
</ul>
);
}
مثال: الاستخدام الصحيح للمفتاح
function ListItem(props) {
// .صحيح! فليس هنالك حاجة لتحديد المفتاح هنا return <li>{props.value}</li>;}
function NumberList(props) {
const numbers = props.numbers;
const listItems = numbers.map((number) =>
// .صحيح! يجب تحديد المفتاح بداخل المصفوفة <ListItem key={number.toString()} value={number} /> );
return (
<ul>
{listItems}
</ul>
);
}
وكقاعدة عامة تحتاج العناصر المُستدعاة من قبل التابع map()
إلى مفاتيح.
ينبغي أن تكون المفاتيح فريدة فقط ضمن العناصر الأشقاء
يجب أن تكون العناصر المستخدمة ضمن المصفوفات فريدة من بين العناصر الأشقاء لها، ولكن ليس بالضرورة أن تكون فريدة بشكل عام، فيمكننا استخدام نفس المفاتيح عند إنتاج مصفوفتين مختلفتين:
function Blog(props) {
const sidebar = ( <ul>
{props.posts.map((post) =>
<li key={post.id}> {post.title}
</li>
)}
</ul>
);
const content = props.posts.map((post) => <div key={post.id}> <h3>{post.title}</h3>
<p>{post.content}</p>
</div>
);
return (
<div>
{sidebar} <hr />
{content} </div>
);
}
const posts = [
{id: 1, title: 'Hello World', content: 'Welcome to learning React!'},
{id: 2, title: 'Installation', content: 'You can install React from npm.'}
];
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(<Blog posts={posts} />);
تعمل المفاتيح كتلميح في React، ولكنها لا تُمرَّر إلى المُكوِّنات. إن احتجت نفس القيمة في مُكوِّناتك فمرّرها بشكل صريح كخاصيّة prop مع استخدام اسم آخر:
const content = posts.map((post) =>
<Post
key={post.id} id={post.id} title={post.title} />
);
في المثال السّابق يستطيع المُكوِّن Post
قراءة props.id
ولكن لا يُمكنه قراءة props.key
.
تضمين التابع map() في JSX
صرّحنا في الأمثلة السّابقة عن المتغير listItems
بشكلٍ مُنفصل وضمّناه في JSX:
function NumberList(props) {
const numbers = props.numbers;
const listItems = numbers.map((number) => <ListItem key={number.toString()} value={number} /> ); return (
<ul>
{listItems}
</ul>
);
}
تسمح JSX بتضمين أي تعبير موجود بين قوسين لذا يمكننا وضع نتيجة التابع map()
بشكل سطري (inline):
function NumberList(props) {
const numbers = props.numbers;
return (
<ul>
{numbers.map((number) => <ListItem key={number.toString()} value={number} /> )} </ul>
);
}
يُؤدّي هذا أحيانًا إلى شيفرةٍ مفهومةٍ بشكلٍ أكبر، ولكن قد يُساء استخدام هذا التنسيق. كما هو الحال في JavaScript لك حريّة القرار إذا ما كان استخراج المتغيرات لسهولة القراءة يستحق العناء. إن كان جسم التابع map()
متداخلًا كثيرًا فمن الأفضل استخراج المُكوِّن.