قبل از خواندن مطالب، مطمین شوید که node js را روی کامپیوتر خود نصب کردهاید.
node --version
حال باید پکیج create-react-app
را با استفاده از دستور زیر نصب کنید:
npm i -g create-react-app
همچنین نصب پکیج pnpm هم بسیار توصیه میشود، چون باعث سریعتر نصب شدن پکیجها میشود و از فضای دیسک بسیار کمتری استفاده میکند که توضیحات آن را در این لینک میتوانید بخوانید.
npm i -g pnpm
در صورتی که از سیستمعامل Ubuntu استفاده میکنید برای این که بتوانید پکیجها را به صورت global نصب کنید، باید کاربر super user باشید:
sudo su
مزیت استفاده از پکیج create-react-app
این است که ما هر بار نیاز به configure کردن پروژه React از ابتدا نداریم و این پکیج به صورت اتوماتیک configuration خاصی از پروژه را میسازد.
create-react-app <app-name>
ساختار پروژهای که با استفاده از این پکیج ساخته میشود به شکل زیر است:
app-name
├── public
| ├── favicon.ico
| ├── index.html
| └── manifest.json
├── src
| ├── App.js
| ├── App.css
| ├── App.test.js
| ├── index.js
| ├── index.css
| └── serviceWorker.js
├── .env
├── .gitignore
├── package.json
└── package.lock.json
این ساختار بسته به نوع پروژه ممکن است مناسب نباشد و برای دسترسی راحتتر به فایلها هنگام انجام پروژه، میتوان ساختارهایی را در نظر گرفت مانند:
src
├── components
├── assets
| ├── images
| ├── audios
| └── videos
├── lib
| ├── api
| ├── utils
| └── validation
├── views
├── App.js
├── App.css
├── App.test.js
├── index.js
├── index.css
└── serviceWorker.js
و
├── package.json
├── public
│ ├── favicon.ico
│ ├── index.html
│ ├── manifest.json
│ └── robots.txt
├── README.md
├── src
│ ├── component
│ ├── index.js
│ ├── serviceWorker.js
│ └── static
│ ├── css
│ │ └── some-css.css
│ └── fonts
│ ├── some-fonts.eot
نمونهای از ساختار بالا را میتوانید در اینجا مشاهده کنید.
پینوشت: برای آشنایی با configure کردن پروژه بدون استفاده از پکیج create-react-app میتوانید لینکهای زیر را ببینید:
package.json فایلی است که برای ذخیره meta-dataها، لیست dependencyها، تعریف script و... از آن استفاده میشود.
این فولدر شامل پکیجهایی است که با استفاده از دستور npm i
یا npm install
نصب شدهاند.
Service Worker یک اسکریپت است که مرورگر شما به صورت مستقل از صفحه وب در background اجرا میکند و ویژگیهایی مثل push notification، background sync و ... را اضافه کرده است.
برای آشنایی بیشتر میتوانیداین لینک را نگاه کنید.
اطلاعات وب اپلیکیشن مثل نام، نویسنده، آیکونهای برنامه، توضیحات و... در این فایل ذخیره میشود.
برای آشنایی بیشتر میتوانیداین لینک را نگاه کنید.
آدرس یا نام فایلهایی که قرار نیست روی گیت آپلود شوند، در این فایل قرار میگیرد.
برای Customize کردن مقادیر بر اساس نوع انتشار برنامه (production, development, ...) و نگه داشتن اطلاعات مهم و حساس مانند پسوورد ها و API Key ها استفاده میشود. این فایل در gitignore قرار داده میشود تا بقیه به اطلاعات آن دسترسی نداشته باشند.
این که واقعا React چیه، داره چیکارمیکنه و چرا باید ازش استفاده کنیم سوالایی هستند که نمیشه به راحتی به اونا جواب داد اما اگه شاید یبار تلاش کنیم React رو تعریف کنیم بتونیم راحت تر باهاش کنار بیایم
حقیقت ماجرا اینه که توی فرمای html زیاد پیش میاد که بخوایم از یه ساختار ساده چندین بار جاهای مختلف استفاده کنیم و این که بخوایم هردفعه عین اون کدارو دوباره بنویسیم به نظر درست نمیاد
اینجاست که پای جاوا اسکریپت به میدون باز میشه چون اگه ما بتونیم یک المان رو فقط یکبار داخل جاوا اسکریپت پیاده سازی کنیم میتونیم بارها و بارها از اون استفاده کنیم
خب این که خیلی خوبه پس مشکلش کجاست؟؟؟
این راه حل درسته که راه حل خوبی به نظر میاد اما چنتا مشکلم داره:
اول این که المانای شبه هم لزوما صددرصد هم شبیه هم نیستن و اگه قرار باشه ما هربار که میخوایم اون المان رو استفاده کنیم مجبور باشیم یه بخشای کوچیکی ازش رو تغییر بدیم این ممکنه خیل برای ما مشکل ساز بشه
<div>
<h1 id="subject">موضوع</h1>
<p id="content">مطلب</p>
</div>
دوما این که همیشه هم المانای ما مثل مثال بالا ساده نیستن و میتونن از چندین تگ تو در توی پیچیده تشکیل بشن که اگه قرار باشه یه object درخت مانند زیادی تو در تو بشه، دیگه کار باهاش اصلا آسون نیست
البته به غیر از مسئله استفاده دوباره از کدها یه مشکل دیگه هم بود که جاوا اسکریپت اونو برای ما حل کرده بود اونم چیزی نبود بجز مطلب پویا
خب الان این همه توضیح دادم که چی؟
هدف من از این حرفا این بود که نشون بدم که درسته که جاوا اسکریپت راه حل این مشکلات ما بود اما همیشه هم راه حل آسونی نبود. در نهایت شاید یه نفر تو این شرایط با خودش بگه کاش یه چیزی بود که همه اینا بود!!!
این که یه چیزی همهی اینا باشه شاید اولش به نظر مسخره بیاد اما به زودی متوجه میشیم که جواب واقعا همینه
اتفاقی که داخل بدنه React میفته اینه که ما میایم و هر جزئی رو یک شیء در نظر میگیریم اما نه شیئی که از به هم اضافه کردن کلی المنت ساخته شده باشه بلکه اون شيء به کمک چند خط html ساخته شده
کد html چند خط قبل به کمک جاوا اسکریپت:
let element = document.createElement("div");
let subject = document.createElement("h1");
let content = document.getElementById("p");
subject.innerText = "موضوع";
subject.id = "subject";
content.innerText = "مطلب";
content.id = "content";
element.appendChild(subject);
element.appendChild(content);
تبدیل اون به تابع برای استفاده آسون تر:
function elem(subject, content){
let elementDivTag = document.createElement("div");
let subjectH1Tag = document.createElement("h1");
let contentPTag = document.getElementById("p");
subjectH1Tag.innerText = subject;
subjectH1Tag.id = "subject";
contentPTag.innerText = content;
contentPTag.id = "content";
elementDivTag.appendChild(subjectH1Tag);
elementDivTag.appendChild(contentPTag);
return elementDivTag;
}
همون کد به کمک React:
function Comp(props) {
const {subject, content} = props;
return (
<div>
<h1 id="subject"> {subject} </h1>
<p id="content"> {content} </p>
</div>
);
}
بازم همون کد به کمک React که درواقع یکم با کلاسای جاوااسکریپت ترکیب شده:
class Comp extends React.Component {
render() {
const { subject, content } = this.props;
return (
<div>
<h1 id="subject"> {subject} </h1>
<p id="content"> {content} </p>
</div>
);
}
}
با این که شاید همه چیز رو از دوتا کد آخر متوجه نشده باشید اما انگار که تو این دوتا کد نسبت به دوتا کد قبلی یه نظم عجیبی وجود داره و این نظم دقیقا از اونجایی میاد که ما میتونیم همزمان و در کنار هم از html و هم از جاوا اسکریپت استفاده کنیم(که البته به این نوع کد JSX
گفته میشه که قراره باهاش بیشتر آشناشیم)
با ساخت پروژه React، یک فایل index.html ساخته میشود که در آن یک المنت با id=root قرار دارد. کاری که ما قرار است انجام دهیم، قرار دادن اجزا در این المنت است. برای اینکار یک فایل index.js در src قرار میگیرد که کد درون آن بدین شکل است:
ReactDOM.render((
<h1> Hello </h1>
), document.getElementById('root'));
با اجرای این کد عبارت Hello در صفحه نشان داده میشود.
کاری که React در آن به ما کمک میکند این است که بجای استفاده از المان های پیشفرض در HTML، میتوانیم خودمون تعدادی Component تعریف کنیم و از آنها استفاده کنیم. با Component ها در ادامه آشنا میشویم. اگر یک Component کلی، مثلا با نام App برای پروژه مان تعریف کنیم، اینگونه میتوانیم آنرا در کد قرار دهیم تا در صفحه نشان داده شود:
ReactDOM.render((
<App/>
), document.getElementById('root'));
نه جاوا اسکریپته، نه کد HTML و نه String.
در واقع داخل {}
فضایی د اختیار ما قرار میگیرد که میتوانیم فقط و فقط یک چیز را در آن قرار دهیم(آن یک چیز میتواند null
یک string
یک root tag
و یا یک آرایه از همین ها باشد). به غیر از این شرط که حتما باید اجراء شود، میتوانیم از تمامی مزایای jsx داخل {}
نیز بهرهمند شویم.
- منظور از root tag، یک تگ بدون هیچ خواهر و یا برادر است که محتوای آن هیچ محدودیتی ندارد
<div id="root">
<div>
<p>علی</p>
</div>
<h4>محسن</h4>
رضا
</div>
به عبارت دیگر داخل آن میتواند:
۱. اسم متغیر باشد.
var tutorialName = "React";
<div>
Let's say hello {tutorialName}!
</div>
۲. تابع صدا زده شود.
function formatName(user) {
return user.firstName + ' ' + user.lastName;
}
const user = {
firstName: 'Harper',
lastName: 'Perez'
};
const element = (
<h1>
Hello, {formatName(user)}!
</h1>
);
۳. و کلی چیز های دیگر ...
const pooya = "پویا";
const condition = true;
const matn = <p dir="rtl"> { pooya } یک متن </p>;
<div>
{
condition ? (
<p>متن اول</p>
) : (
<p>متن دوم</p>
)
}
یک فاصله
{
matn
}
{
["a", "b" ,"c"].map(element => <p> {element} </p>)
}
</div>
مطالب بیشتر در مورد JSX را میتوانید از اینجا بخوانید.
احتمال قبلا با مفهوم DOM آشنا شدید اما اگه نشدید برید تو این لینک اگه لینک قبلی رو مطالعه کردید و یا با از قبل با مفهوم DOM آشنایید باید اینو بهتون بگم که توی React هم در پشت پرده برای ایجاد تغییرات در صفحه از همین توابع استفاده شده است اما با یک تفاوت مهم اونم این که در React در زمانی که یک پارامتر کلیدی ( مثل props یا state که به زودی باهاشون آشنا میشید ) دستخوش تغییر بشند، واکنشی که react نشون میده اینه که به کمک state فعلی یک DOM مجازی (ٰٰٰVirtualDOM) میسازه و اونو با DOM فعلی مقایسه میکنه و تنها تفاوتهاشونو به کمک توابع موجود، در DOM تغییر میده و همین باعث میشه که React از سرعت نسبتا خوبی برخوردار باشه.
Component به ما کمک میکند تا رابط کاربری را به اجزا مستقل و قابل استفاده مجدد تقسیم کنیم و در مورد هر کدام به صورت جداگانه تصمیم بگیریم و دسترسی داشته باشیم.
Componentها به دو صورت میتوانند تعریف شوند:
۱. تابعی
function Welcome() {
return <h1> Hello </h1>;
}
۲. به ارث بردن از کلاس Component
import {Component} from 'react'
class Welcome extends Component {
render() {
return <h1> Hello </h1>;
}
}
حرف اول اسم Component حتما باید بزرگ نوشته شود تا با تگها اشتباه گرفته نشود.
برای Component ها میتوان constructor تعریف کرد و عملیات های اولیه مانند مقدار دهی ها را در آن انجام داد. کد زیر مثالی برای استفاده از constructor است:
import {Component} from 'react'
class Welcome extends Component {
constructor(){
this.name="Mohammad"
}
render() {
return <h1> Hello {this.name} </h1>;
}
}
پس از تعریف کردن یک Component، در Component های دیگر میتوانیم از آن استفاده کنیم:
import {Component} from 'react'
class Welcome extends Component {
render() {
return <h1> Hello {this.props.name} </h1>;
}
}
class MainPage extends Component {
render() {
return (
<Welcome name="Asghar"/>
<Welcome name="Showkat"/>
<Welcome name="Fahime"/>
)
}
}
ReactDOM.render((
<MainPage/>
), document.getElementById('root'));
در مورد props استفاده شده در کد بالا در بخش بعدی توضیح خواهیم داد.
توصیه میشود که صفحه های مختلف و همچنین اجزای مختلف هر صفحه را در Component های متفاوت پیاده سازی و استفاده کنید.
برای استفاده از یک Component در Component های دیگر، باید آنرا در مبدا export و در Component های مقصد import کنیم:
import {Component} from 'react'
class Welcome extends Component {
render() {
return <h1> Hello {this.props.name} </h1>;
}
}
export default Welcome
import {Component} from 'react'
import Welcome from './Welcome'
class MainPage extends Component {
render() {
return (
<div>
<Welcome name="Asghar"/>
<Welcome name="Showkat"/>
<Welcome name="Fahime"/>
</div>
)
}
}
به Componentها میتوان یک object پاس داد که به آن (Props (Properties
گفته میشود. در Componentهای اول با {<props.<prop-name}
میتوانیم به prop مورد نظر دسترسی پیدا کنیم. اما در Component نوع دوم باید از {<this.props.<prop-name}
استفاده کنیم.
برای Type-Checking میتوان نوع prop را با PropTypes مشخص کرد.
import {Component} from 'react'
import PropTypes from 'prop-types';
class MainPage extends Component {
render(){
return <Greeting name="Ali"/>
}
}
class Greeting extends Component {
render(){
return (
<h1>Hello, {this.props.name}</h1>
);
}
}
Greeting.propTypes = {
name: PropTypes.string
};
با استفاده از props میتوانیم توابعی که در یک Component تعریف شده اند را نیز به Component های دیگر منتقل کرده و از آنها استفاده کنیم. برای مثال اگر بخواهیم تغییری در Component های بالاتر ایجاد کنیم، میتوانیم تابعی برای این تغییر را از آن Component ها به پایین از طریق props انتقال دهیم.
import {Component} from 'react'
class MainPage extends Component {
clicked = () => {
console.log("Button Clicked")
}
render() {
return <SomeComponent func={this.clicked}/>;
}
}
class SomeComponent extends Component {
constructor(props){
super(props)
this.props.func()
}
render() {
return <div>Doesn't Matter</div>;
}
}
اگر در Component از props استفاده شود و constructor را نیز پیاده سازی کرده باشیم، در ورودی constructor باید props گرفته شود و در خط اول آن به super پاس داده شود.
در React یک Component نباید props خودش را تغییر دهد.
تا اینجای کار، یاد گرفتیم که صفحه هایی بسازیم و در آنها اجزایی قرار دهیم که با استفاده از props، میتوانستیم این اجزا و چگونگی قرار گرفتن آنها در صفحه را مشخص کنیم. امروزه سایت های زیادی را مشاهده میکنیم که بخش های مختلف سایت، طبق عملیات های کاربر تغییر میکنند و به اصطلاح interactive اند.
از این رو احساس نیاز ما برای اینکه صفحه مان حالات مختلفی داشته باشد که بنابر اطلاعات و عمل های مختلف کاربر تغییر کنند. در React، state ها این وظیفه را بر عهده دارند. state را میتوان به شکل یک object در نظر گرفت که وضعیت صفحه را مشخص میکند. برای مثال اگر بخواهیم یک جزء لامپ داشته باشیم که روشن یا خاموش باشد، در state متغیری به نام isLit در state میگیریم و به آن مقادیر True و False میدهیم.
به عنوان مثالی ساده، کد زیر هر چیزی که در InputField تایپ کنیم را در پاراگراف زیرش نیز نشان میدهد:
import {Component} from 'react'
class Text extends Component {
state={
text:"Default Value",
}
render() {
return (
<div>
<input type="text" onChange={ (e) => { this.setState({ text:e.target.value }) } } value={ this.state.text } />
<p>{ this.state.text }</p>
</div>
);
}
}
برای استفاده از یک state از
this.state.<state-name>
و برای مقدار دهی به یک state در برنامه، ازthis.setState({<state-name>:<state-value>})
استفاده میشود.
علاوه بر مقدار دهی اولیه state در بدنه Component، در constructor نیز میتوان با this.state={<state-name>:<state-value>}
به آن مقدار دهی کرد. اما در بخش های دیگر باید از setState استفاده کرد.
دقت کنید که پس از استفاده از setState، مقدار state در همان لحظه تغییر نمیکند، در بخش بعدی که LifeCycle است با طرز کار state بیشتر آشنا میشویم.
Component های از نوع function، از مفهوم Hooks استفاده میکنند که به نسبت ساده و تمیز است. مثالی از آن را در کد زیر میبینید:
import React, { useState } from 'react';
function Example() {
// Declare a new state variable, which we'll call "count"
const [count, setCount] = useState(0);
return (
<div>
<p>You clicked {count} times</p>
<button onClick={() => setCount(count + 1)}>
Click me
</button>
</div>
);
}
برای استفاده از Hooks باید از useState استفاده کنیم و به عنوان ورودی به آن مقدار اولیه state را بدهیم. این تابع دو خروجی میدهد که اولی همان متغییر یا state مان است و دومی تابعیست که با آن میتوانیم state مان را تغییر دهیم (مانند setState).
برای آشنایی بیشتر با Hooks میتوانید از این لینک استفاده کنید.
هر Component در React یک LifeCycle دارد که به سه فاز Mounting، Updating و Unmounting تقسیم میشود. در هر فاز، تعدادی از توابع در Component صدا زده میشوند که وظیفه ی ما قرار دادن کدمان در تابع مناسب برای کاریست که میخواهیم انجام دهیم.
در فاز Mount توابع زیر به ترتیب اجرا میشوند:
- constructor : در این تابع مقدار دهی اولیه به state انجام داده میشود. ورودی تابع props است بنابراین میتوانیم مقدار دهی را با استفاده از props انجام دهیم. در ابتدای تابع باید super(props) را صدا کنیم.
- render : خروجی این تابع صفحه ایست که میخواهیم نشان داده شود.
- componentDidMount : در این تابع کارهایی که نیاز به render شدن صفحه مثلا دسترسی به DOM را دارند، انجام میدهیم.
اگر تغییری در props یا state ایجاد شود، فاز Updating شروع میشود و توابع زیر به ترتیب اجرا میشوند:
- shouldComponentUpdate : یک Boolean خروجی میدهد بدین معنا که آیا بقیه توابع Updating صدا زده بشوند یا که فاز Updating متوقف شود. در صورت عدم پیاده سازی این تابع خروجی آن همیشه True خواهد بود.
- render : خروجی این تابع صفحه ایست که میخواهیم نشان داده شود.
- getSnapshotBeforeUpdate : در این تابع به props و state های قبل از Update دسترسی داریم و میتوانیم از آنها استفاده کنیم. ورودی های تابع را میتوان به صورت prevProps و prevState در نظر گرفت. در صورت پیاده سازی این تابع، تابع componentDidUpdate نیز باید پیاده سازی شود.
- componentDidUpdate : در این تابع عملیات های بعد از render شدن صفحه را انجام میدهیم.
وقتی که Component مورد نظر از DOM خارج میشود (دیگر در صفحه حاظر نیست)، فاز Unmounting اجرا میشود و تابع componentWillUnmount صدا زده میشود.
هنگامی که از functional Component ها و Hooks استفاده میکنیم، بجای تمام توابع componentDidMount، componentDidUpdate و componentWillUnmount میتوانیم تنها از useEffect استفاده کنیم:
import { useState, useEffect } from "react";
export default function useFetch(url) {
const [data, setData] = useState([]);
useEffect(() => {
fetch(url)
.then(response => response.json())
.then(data => setData(data));
}, []);
return data;
}
کد بالا یک url به عنوان ورودی میگیرد و اطلاعات دریافتی از آن را خروجی میدهد. گفتیم که تابع useEffect در هر 3 فاز اجرا میشود، اما ما فقط میخواهیم در فاز اول اجرا شود، به همین دلیل ورودی دومش را آرایه ای خالی داده ایم. ورودی دوم useEffect در حالت کلی state هایی است که میخواهیم با تغییر آنها فاز Updating انجام شود.
![](/mirshaf/Web-Workshop/raw/master/React/LifeCycle.png)
تمام event هایی که برای تگهای داخل فرم html تعریف شده، در React نیز برای این تگ ها تعریف شده است، فقط با این تفاوت که در اینجا نام آن ها باید به صورت lowerCamelCase بکار برده شود
- در حالت کلی در React تمام prop ها(attribute ها)ی یک element یا component به صورت lowerCamelCase در دسترس اند و میتوان آنها را به کمک یک
{}
و یا در صورت امکان یک string مقدار دهی کرد
برای مقدار دهی آنها باید از {}
استفاده شود:
import React, { Component } from 'react';
class Comp extends Component {
constructor(props) {
super(props);
this.state = {
value: 0,
};
}
setValueByTargetValueFunc(e) {
this.setState({
value: e.target.value
})
}
render() {
const { value } = this.state;
return (
<div>
<input
type="text"
value={value}
onChange={this.setValueByTargetValueFunc.bind(this)}
/>
<input
type="text"
value={value}
onChange={this.setValueByTargetValueFunc.bind(this)}
/>
{
value
}
</div>
);
}
}
export default Comp;
در صورتی که بخواهیم از یک تابع چند بار استفاده کنیم برای این که نخواهیم هر بار عمل bind را انجام دهیم میتوانیم این کار را در همان ابتدا انجام دهیم
import React, { Component } from 'react';
class Comp extends Component {
constructor(props) {
super(props);
this.state = {
value: 0,
};
this.setValueByTargetValue = function(e) {
this.setState({
value: e.target.value
})
}.bind(this)
}
render() {
const { value } = this.state;
return (
<div>
<input
type="text"
value={value}
onChange={this.setValueByTargetValue}
/>
<input
type="text"
value={value}
onChange={this.setValueByTargetValue}
/>
{
value
}
</div>
);
}
}
export default Comp;
در روشی دیگر نیز میتوان از arrow function ها نیز استفاده نمود زیرا در این نوع توابع this به صورت خودکار به تابع bind میشود.
import React, { Component } from 'react';
class Comp extends Component {
constructor(props) {
super(props);
this.state = {
value: 0,
};
}
setStateByTargetValue = (state) => element => {
this.setState({
[state]: element.target.value,
})
};
setValueByTargetValue = e => {
this.setState({
value: e.target.value
})
};
render() {
const { value } = this.state;
return (
<div>
<input
type="text"
value={value}
onChange={this.setStateByTargetValue("value")}
/>
<input
type="text"
value={value}
onChange={this.setValueByTargetValue}
/>
<input
type="text"
value={value}
onChange={(e) => this.setState({value: e.target.value})}
/>
{
value
}
</div>
);
}
}
export default Comp;
مثال های بیشتر از فرم ها را از اینجا مشاهده کنید.
در React فرمها تفاوت عمدهای با فرم های عادی ندارند اما دو ویژگی جزئي در آنها وجود دارد: ۱.استفاده از htmlFor به جای for برای label ها ۲.ویژگی دوم در گذشته نیز وجود داشت اما در React کار برد زیادی دارد و آن چیزی نیست بجز تابع preventDefault
- علت استفاده از این تابع نیز جلوگیری از reload شدن صفحه میباشد
import React from "react";
class NameForm extends React.Component {
constructor(props) {
super(props);
this.state = {value: ''};
this.handleChange = this.handleChange.bind(this);
this.handleSubmit = this.handleSubmit.bind(this);
}
handleChange(event) {
this.setState({value: event.target.value});
}
handleSubmit(event) {
alert('A name was submitted: ' + this.state.value);
event.preventDefault();
}
render() {
return (
<form onSubmit={this.handleSubmit}>
<label htmlFor="input-text">
Name:
</label>
<input type="text" value={this.state.value} onChange={this.handleChange} id="input-text"/>
<input type="submit" value="Submit" />
</form>
);
}
}
#Single Page Application(SPA) and React Router
ریاکت یک ابزار قدرتمند برای ساخت SPA میباشد اما SPA به چه معناست؟
تفاوت عمده SPA ها با دیگر سایت ها در عدم load کردن دوباره صفحه است که این تجربه بهتری را برای کاربر به همراه دارد و در واقع صفحه وب شما مانند یک desktop application به نظر خواهد آمد.
اگر سایت شما زیاد بزرگ نباشد شاید بتوانید به سادگی آن را به صورت یک SPA پیاده سازی کنید اما وقتی کمی سایت بزرگتر شود و تعداد Component ها زیاد شود این باعث میشود که نتوان به سادگی همهی آن ها را در کنار هم نشان داد.
برای چنین مواقعی ابزاری در React وجود دارد که این موضوع را برای ما حل خواهد کرد و آن هم react-router میباشد.
ما به کمک این ازار میتوانیم url های مختلف را به Component هایمان لینک کنیم که این کار به صورت زیر انجام میشود
import React, {Component} from "react";
import {BrowserRouter, Route} from "react-router-dom";
import SignIn from "./SignIn";
import SignUp from "./SignUp";
import ForgotPassword from "./ForgotPassword";
import Main from "./Main";
import Home from "./Home";
import Posts from "./Posts";
class MainRoute extends Component {
render() {
return (
<BrowserRouter>
<div> صفحه ها </div>
<Route path="/sign-in" component={SignIn} />
<Route path="/sign-up" component={SignUp} />
<Route path="/forgot-password" component={ForgotPassword} />
<Route exact path="/" component={Main} />
<Route exact path="/home" component={Home} />
<Route path="/home/posts" component={Posts} />
</BrowserRouter>
);
}
}
در مثال های بالا به ازای هر url یک کامپوننت خاص render میشود که به کمک path آدرس آن مشخص شده است
- کلمه exact برای این است که component فقط با همان url نشان داده شود و در غیر این صورت کامپوننت برای url های مکمل نیز نشان داده خواهد شد
در مثال بالا همانطور که میبینید از BrowserRouter به عنوان پدر تام Route ها استفاده شده است و باید این را بدانید که BrowserRouter کافیست تنها پدر و یا یکی از اجداد یک Route باشد تا Route به درستی کارکند بنابر این کافیست در ابتدا فقط یکبار پدر تمام Component ها را به کمک آن Wrap کنیم و سپس در هر جایی که بخواهیم میتوانیم از Route استفاده کنیم
#Higher Order Component(HOC)
گاهی اوقات ما یک Component را به کمک Component دیگری Wrap میکنیم و به کمک آن تعدادی prop به props آن اضافه خواهیم کرد به این Component که در نقش Wrapper ضاهر میشود، Higher Order Component یا به اختصار HOC گفته میشود.
- در حالت کلی HOC ها معمولا در نقش یک تابع ظاهر میشوند که Component اولیه را به عنوان ورودی میگیرند و آن را به صورت Wrap شده خروجی میدهند
یکی ازاین HOC ها withRouter میباشد که بعد از Wrap کردن Component یک field با نام history به props آن اضافه میکنند که آن field توابع مهمی را در اختیار ما قرا میده که به کمک آن ها میتوانیم با url در ارتباط باشیم.
به عنوان مثال تابع push یک url میگیرد و صفحه را به آدرس آن redirect میکند.
import React, {Component} from "react";
import { withRouter } from "react-router";
class BrowserRouterChild extends Component {
render() {
return (
<input type="text" value="redirect" onClick={() => this.props.history.push("/home")}/>
);
}
}
export default withRouter(BrowserRouterChild);
- فقط به این موضوع دقت کنید که از withRouter تنها در Component هایی میتوان استفاده کرد از قرزندان ویا نوادگان BrowserRouter باشند