مطور الويب، مهما كان ماهرا أو محترفا، قد يقع في أخطاء بسيطة في بداية مشواره مع تقنية معينة لها خصائص معينة وقواعد يجب احترامها.
مكتبة React.js واحدة من تقنيات تطوير الويب التي ذاع صيتها في السنوات الأخيرة وأصبح استخدامها شائعا بشكل كبير من قبل مطوري واجهات الويب الأمامية.
سأحاول في هذا الموضوع جرد أهم وأكثر الأخطاء الشائعة التي يواجهها هؤلاء المطورين مع React.js، وهي على بساطتها إلا أنها تحظى بكم هائل من الأسئلة في StackOverflow، ما جعلني أحاول جمعها في هذا المقال لتكون مرجعا لكل مطور يفكر في بدء مساره مع هذه المكتبة.
1. تسمية المكونات
بعض المطورين يقومون بتسمية المكونات بأسماء تبدأ بحروف صغيرة (lower case)، وهذا خطأ فادح لأن React سيعتبر هذا المكون عند استدعائه بمثابة وسم HTML اعتيادي وليس React component.
class mycomponent extends Component { /* ... */ }
1
2
3
class mycomponent extends Component {
/* ... */
}
عند محاولة عرض المكون mycomponent أعلاه، سنحصل على رسالة تحذيرية ولن يتم عرضه حتى نقوم بتغيير اسمه ليبدأ بحرف كبير (upper case) كما يلي :
class Mycomponent extends Component { /* ... */ }
1
2
3
class Mycomponent extends Component {
/* ... */
}
وفي حالة استخدام أسماء محجوزة لوسوم HTML مثل button، فإنه عند استدعائه سيقوم React بعرض عنصر ال HTML وإهمال مكون React بشكل كلي.
لذلك في جميع الحالات، لا بد أن نبدأ أسماء المكونات بحروف كبيرة.
2. استخدام React.PropTypes
في السابق كان الكائن PropTypes ضمن الكائن الكبير React، فكان يستخدم بهذه الشكل React.PropTypes.
ولكن في النسخ الأخيرة تم تحويل PropTypes إلى حزمة مستقلة يتم تثبيتها عن طريق مدير الحزم npm ثم استيرادها عن الحاجة بهذه الطريقة :
import React from 'react'; import PropTypes from 'prop-types'; class MyComponent extends React.Component { render() { ... } } MyComponent.propTypes = { myFunction: PropTypes.func.isRequired }
1
2
3
4
5
6
7
8
9
10
11
12
import React from 'react';
import PropTypes from 'prop-types';
class MyComponent extends React.Component {
render() {
...
}
}
MyComponent.propTypes = {
myFunction: PropTypes.func.isRequired
}
3. تمرير الأعداد كما نمرر النصوص
في React.js يمكن تمرير النصوص للمكونات باستخدام props بهذه الكيفية :
<User name="Ahmed" />
1
<User name="Ahmed" />
ولكن عند تمرير الأعداد فالمسألة تختلف قليلا، حيث أن تمرير العدد بين مزدوجتين سيجعل المكون يعتبر نصا، وبالتالي لن نستطيع تطبيق العمليات الخاصة بالأعداد عليه، مما قد يتسبب في ظهور أخطاء غير متوقعة.
الطريقة الصحيحة لتمرير الخصائص العددية للمكونات في React.js هي وضعها بين الأقواس المعقوفة :
<User age={30} />
1
<User age={30} />
4. الخلط بين الأقواس المعقوفة والأقواس العادية
عادة ما تستخدم الأقواس العادية (…) لإرجاع محتوى المكونات في الدالة render :
render() { return ( <header className="App-header"> <img src={logo} className="App-logo" alt="logo" /> <h1 className="App-title">Welcome to React</h1> </header> ); }
1
2
3
4
5
6
7
8
render() {
return (
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<h1 className="App-title">Welcome to React</h1>
</header>
);
}
وتستخدم كذلك هذه الأقواس بشكل كبير في الدوال السهمية التي شرحناها في مقال سابق.
ولكن المبتدئين في أحيان كثيرة يقعون عرضة للخلط بين هذا النوع من الأقواس والأقواس المعقوفة، ما يؤدي لمثل هذه الخطأ :
// خطأ!! render() { return { <header className="App-header"> <img src={logo} className="App-logo" alt="logo" /> <h1 className="App-title">Welcome to React</h1> </header> }; }
1
2
3
4
5
6
7
8
9
10
// خطأ!!
render() {
return {
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<h1 className="App-title">Welcome to React</h1>
</header>
};
}
في جافاسكريبت، الأقواس المعقوفة بجانب كلمة return تعني بأنه سيتم إرجاع كائن جافاسكريبت من نوع json، ولكن في هذه الحالة نحن نقوم بإرجاع كود JSX الذي يتم تحويله في مرحلة البناء (Build step) لدوال جافاسكريبت عادية، وهذا يخالف توقعات مفسر الجافاسكريبت مما يتسبب أخيرا في ظهور رسالة الخطأ.
5. الخلط بين الكائن state والكائنات الأخرى داخل المكون
في React.js، يمكن إنشاء كائن محلي اسمه state والوصول إليه انطلاقا من الكائن this الذي يشير لكلاس المكون :
class User extends React.Component { state = { name: "Ahmed", }; render() { return `Name: ${this.state.name}`; } }
1
2
3
4
5
6
7
8
9
class User extends React.Component {
state = {
name: "Ahmed",
};
render() {
return `Name: ${this.state.name}`;
}
}
المخرج النهائي في هذا الكود سيكون “Name: Ahmed”.
إلى جانب الكائن state، يمكننا إنشاء كائنات أخرى داخل كلاس المكون والوصول إليها بنفس الكيفية من خلال this :
class User extends React.Component { user = { name: "Ahmed", }; render() { return `Name: ${this.user.name}`; } }
1
2
3
4
5
6
7
8
9
class User extends React.Component {
user = {
name: "Ahmed",
};
render() {
return `Name: ${this.user.name}`;
}
}
والنتيجة ستكون هي نفسها “Name: Ahmed”.
ولكن هناك فرق كبير بين الكائن المسمى state وأي كائن أو خاصية أخرى لمكونات React. هذا الفرق يتمثل في كون state يتم استخدامه وإدارته من طرف React لعرض وتحديث حالة المكون (واجهة المستخدم) كلما طرأ تغيير عليه. والطريقة الوحيدة لإحدات تغيير على الكائن state هي عن طريق الدالة ()setState.
أما الكائنات الأخرى مثل user في حالتنا، فهي مجرد خصائص عادية للكلاس ولا تلعب أي دور في تحديث حالة المكون مهما غيرنا من قيمها، ولا تعرض قيمها الجديدة في واجهة المستخدم إلا بعد تحديث المكون عن طريق ()setState.
6. قيمة this في وظائف الأحداث
في React.js، جميع الوظائف التي يتم استدعاؤها داخل الدالة render تربط الكائن this بداخلها بكلاس المكون الذي يضمها. ولكن الوظائف التي يتم استدعاؤها عند أحداث معينة (مثلا onClick أو onChange) استثناء لهذه القاعدة.
دعونا نرى المثال التالي :
class MyComponent extends React.Component { whoIsThis() { console.log(this); } render() { this.whoIsThis(); // MyComponent return <button onClick={this.whoIsThis}>Who is This</button>; // undefined } }
1
2
3
4
5
6
7
8
9
10
11
class MyComponent extends React.Component {
whoIsThis() {
console.log(this);
}
render() {
this.whoIsThis(); // MyComponent
return <button onClick={this.whoIsThis}>Who is This</button>; // undefined
}
}
في الشفرة أعلاه، الدالة ()whoIsThis التي يتم استدعاؤها من داخل render يؤشر this بداخلها إلى MyComponent، ولكن عندما نقوم باستدعائها عند النقر على الزر button فإن this يؤشر إلى undefined، وهذا مشكل كبير لأننا عادة نحتاج لإستخدام this على أنه المكون نفسه في هذه الأحداث، وخاصة عند التعامل مع this.state أو this.props.
ما هو الحل إذن ؟
في الحقيقة هناك أكثر من حل لهذا المشكل، ولكن أفضلها بحسب رأيي هو الإستعانة بالدوال السهمية التي تحفظ للكائن this قيمته مهما اختلفت ظروف استدعائها.
إذن الحل هنا يكمن في تحويل ()whoIsThis إلى دالة سهمية، لا أقل ولا أكثر، بعدها تعود إلى الأمور إلى نصابها:
class MyComponent extends React.Component { whoIsThis = () => { console.log(this); } render() { this.whoIsThis(); // MyComponent return <button onClick={this.whoIsThis}>Who is This</button>; // MyComponent } }
1
2
3
4
5
6
7
8
9
10
11
class MyComponent extends React.Component {
whoIsThis = () => {
console.log(this);
}
render() {
this.whoIsThis(); // MyComponent
return <button onClick={this.whoIsThis}>Who is This</button>; // MyComponent
}
}
النهاية
هذه جملة الأخطاء التي يواجهها المبتدؤون بكثرة في بداية مشوارهم مع React.js، وقد واجهت أنا أيضا بعضا منها في أيامي الأولى مع هذه المكتبة أتمنى أن تستفيدوا منها وتتجنبوها في مشاريعكم القادمة، وسأحاول تحديث هذه القائمة باستمرار كلما صادفت مشكلا آخر من المشاكل البسيطة التي تواجه المبتدئين مع مكتبة React.js.
مكتبة React.js واحدة من تقنيات تطوير الويب التي ذاع صيتها في السنوات الأخيرة وأصبح استخدامها شائعا بشكل كبير من قبل مطوري واجهات الويب الأمامية.
سأحاول في هذا الموضوع جرد أهم وأكثر الأخطاء الشائعة التي يواجهها هؤلاء المطورين مع React.js، وهي على بساطتها إلا أنها تحظى بكم هائل من الأسئلة في StackOverflow، ما جعلني أحاول جمعها في هذا المقال لتكون مرجعا لكل مطور يفكر في بدء مساره مع هذه المكتبة.
1. تسمية المكونات
بعض المطورين يقومون بتسمية المكونات بأسماء تبدأ بحروف صغيرة (lower case)، وهذا خطأ فادح لأن React سيعتبر هذا المكون عند استدعائه بمثابة وسم HTML اعتيادي وليس React component.
class mycomponent extends Component { /* ... */ }
1
2
3
class mycomponent extends Component {
/* ... */
}
عند محاولة عرض المكون mycomponent أعلاه، سنحصل على رسالة تحذيرية ولن يتم عرضه حتى نقوم بتغيير اسمه ليبدأ بحرف كبير (upper case) كما يلي :
class Mycomponent extends Component { /* ... */ }
1
2
3
class Mycomponent extends Component {
/* ... */
}
وفي حالة استخدام أسماء محجوزة لوسوم HTML مثل button، فإنه عند استدعائه سيقوم React بعرض عنصر ال HTML وإهمال مكون React بشكل كلي.
لذلك في جميع الحالات، لا بد أن نبدأ أسماء المكونات بحروف كبيرة.
2. استخدام React.PropTypes
في السابق كان الكائن PropTypes ضمن الكائن الكبير React، فكان يستخدم بهذه الشكل React.PropTypes.
ولكن في النسخ الأخيرة تم تحويل PropTypes إلى حزمة مستقلة يتم تثبيتها عن طريق مدير الحزم npm ثم استيرادها عن الحاجة بهذه الطريقة :
import React from 'react'; import PropTypes from 'prop-types'; class MyComponent extends React.Component { render() { ... } } MyComponent.propTypes = { myFunction: PropTypes.func.isRequired }
1
2
3
4
5
6
7
8
9
10
11
12
import React from 'react';
import PropTypes from 'prop-types';
class MyComponent extends React.Component {
render() {
...
}
}
MyComponent.propTypes = {
myFunction: PropTypes.func.isRequired
}
3. تمرير الأعداد كما نمرر النصوص
في React.js يمكن تمرير النصوص للمكونات باستخدام props بهذه الكيفية :
<User name="Ahmed" />
1
<User name="Ahmed" />
ولكن عند تمرير الأعداد فالمسألة تختلف قليلا، حيث أن تمرير العدد بين مزدوجتين سيجعل المكون يعتبر نصا، وبالتالي لن نستطيع تطبيق العمليات الخاصة بالأعداد عليه، مما قد يتسبب في ظهور أخطاء غير متوقعة.
الطريقة الصحيحة لتمرير الخصائص العددية للمكونات في React.js هي وضعها بين الأقواس المعقوفة :
<User age={30} />
1
<User age={30} />
4. الخلط بين الأقواس المعقوفة والأقواس العادية
عادة ما تستخدم الأقواس العادية (…) لإرجاع محتوى المكونات في الدالة render :
render() { return ( <header className="App-header"> <img src={logo} className="App-logo" alt="logo" /> <h1 className="App-title">Welcome to React</h1> </header> ); }
1
2
3
4
5
6
7
8
render() {
return (
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<h1 className="App-title">Welcome to React</h1>
</header>
);
}
وتستخدم كذلك هذه الأقواس بشكل كبير في الدوال السهمية التي شرحناها في مقال سابق.
ولكن المبتدئين في أحيان كثيرة يقعون عرضة للخلط بين هذا النوع من الأقواس والأقواس المعقوفة، ما يؤدي لمثل هذه الخطأ :
// خطأ!! render() { return { <header className="App-header"> <img src={logo} className="App-logo" alt="logo" /> <h1 className="App-title">Welcome to React</h1> </header> }; }
1
2
3
4
5
6
7
8
9
10
// خطأ!!
render() {
return {
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<h1 className="App-title">Welcome to React</h1>
</header>
};
}
في جافاسكريبت، الأقواس المعقوفة بجانب كلمة return تعني بأنه سيتم إرجاع كائن جافاسكريبت من نوع json، ولكن في هذه الحالة نحن نقوم بإرجاع كود JSX الذي يتم تحويله في مرحلة البناء (Build step) لدوال جافاسكريبت عادية، وهذا يخالف توقعات مفسر الجافاسكريبت مما يتسبب أخيرا في ظهور رسالة الخطأ.
5. الخلط بين الكائن state والكائنات الأخرى داخل المكون
في React.js، يمكن إنشاء كائن محلي اسمه state والوصول إليه انطلاقا من الكائن this الذي يشير لكلاس المكون :
class User extends React.Component { state = { name: "Ahmed", }; render() { return `Name: ${this.state.name}`; } }
1
2
3
4
5
6
7
8
9
class User extends React.Component {
state = {
name: "Ahmed",
};
render() {
return `Name: ${this.state.name}`;
}
}
المخرج النهائي في هذا الكود سيكون “Name: Ahmed”.
إلى جانب الكائن state، يمكننا إنشاء كائنات أخرى داخل كلاس المكون والوصول إليها بنفس الكيفية من خلال this :
class User extends React.Component { user = { name: "Ahmed", }; render() { return `Name: ${this.user.name}`; } }
1
2
3
4
5
6
7
8
9
class User extends React.Component {
user = {
name: "Ahmed",
};
render() {
return `Name: ${this.user.name}`;
}
}
والنتيجة ستكون هي نفسها “Name: Ahmed”.
ولكن هناك فرق كبير بين الكائن المسمى state وأي كائن أو خاصية أخرى لمكونات React. هذا الفرق يتمثل في كون state يتم استخدامه وإدارته من طرف React لعرض وتحديث حالة المكون (واجهة المستخدم) كلما طرأ تغيير عليه. والطريقة الوحيدة لإحدات تغيير على الكائن state هي عن طريق الدالة ()setState.
أما الكائنات الأخرى مثل user في حالتنا، فهي مجرد خصائص عادية للكلاس ولا تلعب أي دور في تحديث حالة المكون مهما غيرنا من قيمها، ولا تعرض قيمها الجديدة في واجهة المستخدم إلا بعد تحديث المكون عن طريق ()setState.
6. قيمة this في وظائف الأحداث
في React.js، جميع الوظائف التي يتم استدعاؤها داخل الدالة render تربط الكائن this بداخلها بكلاس المكون الذي يضمها. ولكن الوظائف التي يتم استدعاؤها عند أحداث معينة (مثلا onClick أو onChange) استثناء لهذه القاعدة.
دعونا نرى المثال التالي :
class MyComponent extends React.Component { whoIsThis() { console.log(this); } render() { this.whoIsThis(); // MyComponent return <button onClick={this.whoIsThis}>Who is This</button>; // undefined } }
1
2
3
4
5
6
7
8
9
10
11
class MyComponent extends React.Component {
whoIsThis() {
console.log(this);
}
render() {
this.whoIsThis(); // MyComponent
return <button onClick={this.whoIsThis}>Who is This</button>; // undefined
}
}
في الشفرة أعلاه، الدالة ()whoIsThis التي يتم استدعاؤها من داخل render يؤشر this بداخلها إلى MyComponent، ولكن عندما نقوم باستدعائها عند النقر على الزر button فإن this يؤشر إلى undefined، وهذا مشكل كبير لأننا عادة نحتاج لإستخدام this على أنه المكون نفسه في هذه الأحداث، وخاصة عند التعامل مع this.state أو this.props.
ما هو الحل إذن ؟
في الحقيقة هناك أكثر من حل لهذا المشكل، ولكن أفضلها بحسب رأيي هو الإستعانة بالدوال السهمية التي تحفظ للكائن this قيمته مهما اختلفت ظروف استدعائها.
إذن الحل هنا يكمن في تحويل ()whoIsThis إلى دالة سهمية، لا أقل ولا أكثر، بعدها تعود إلى الأمور إلى نصابها:
class MyComponent extends React.Component { whoIsThis = () => { console.log(this); } render() { this.whoIsThis(); // MyComponent return <button onClick={this.whoIsThis}>Who is This</button>; // MyComponent } }
1
2
3
4
5
6
7
8
9
10
11
class MyComponent extends React.Component {
whoIsThis = () => {
console.log(this);
}
render() {
this.whoIsThis(); // MyComponent
return <button onClick={this.whoIsThis}>Who is This</button>; // MyComponent
}
}
النهاية
هذه جملة الأخطاء التي يواجهها المبتدؤون بكثرة في بداية مشوارهم مع React.js، وقد واجهت أنا أيضا بعضا منها في أيامي الأولى مع هذه المكتبة أتمنى أن تستفيدوا منها وتتجنبوها في مشاريعكم القادمة، وسأحاول تحديث هذه القائمة باستمرار كلما صادفت مشكلا آخر من المشاكل البسيطة التي تواجه المبتدئين مع مكتبة React.js.