در قسمت قبل با معرفی نوع props توسط TypeScript، مجبور به تکمیل اجباری تک تک آنها شدیم؛ اما در React میتوان props را به صورت اختیاری و یا با مقادیری پیشفرض نیز تعریف کرد.
روش تعیین props پیشفرض توسط TypeScript
اگر بخواهیم توسط روشهای خود React، مقادیر پیشفرض props را تعیین کنیم، میتوان از defaultProps به صورت زیر با تعریف یک شیء جاوا اسکریپتی از پیش مقدار دهی شده، استفاده کرد:
اما در حالت استفادهی از TypeScript و یا حتی نگارش ES6 آن (React در حالت پیشفرض آن)، میتوان مقادیر پیشفرض props را با مقدار دهی مستقیم متغیرهای حاصل از Object Destructuring آن، تعیین کرد:
در اینجا هر متغیری که با مقداری پیشفرض، مقدار دهی شده باشد، اختیاری در نظر گرفته شده و اگر دارای مقدار پیشفرضی نباشد، باید به صورت اجباری در حین تعریف المان این کامپوننت، ذکر شود.
در این حالت انتظار داریم که در حین استفاده و تعریف المان کامپوننت Head، اگر برای مثال ویژگی isActive را ذکر نکردیم، کامپایلر TypeScript خطایی را گزارش نکند؛ که اینطور نیست:
هنوز هم در اینجا میتوان خطای عدم تعریف خاصیت isActive را مشاهده کرد. برای رفع این مشکل، به صورت زیر عمل میکنیم:
در حین تعریف یک type، اگر خاصیتی با علامت ? ذکر شود، به معنای اختیاری بودن آن است. همچنین در اینجا مقدار پیشفرض title را هم حذف کردهایم تا تعریف آن اجباری شود. بنابراین در typeها، تمام خواص اجباری هستند؛ مگر اینکه توسط ? به صورت اختیاری تعریف شوند. این مورد هم مزیتی است که در ابتدای طراحی props یک کامپوننت، باید در مورد اختیاری و یا اجباری بودن آنها بیشتر فکر کرد. همچنین نیازی به استفاده از روشهای غیراستانداردی مانند Head.defaultProps خود React نیست. ذکر مقدار پیشفرض متغیرهای حاصل از Object Destructuring، جزئی از جاوااسکریپت استاندارد است و یا مشخص سازی خواص اختیاری در TypeScript، فقط مختص به پروژههای React نیست و در همه جا به همین شکل کاربرد دارد.
اکنون با تعریف isActive?: boolean، دیگر شاهد نمایش خطایی در حین تعریف المان Head، بدون ذکر خاصیت isActive، نخواهیم بود.
تعریف انواع و اقسام نوعهای props
تا اینجا نوعهای سادهای مانند string و boolean و همچنین نحوهی تعریف اجباری و اختیاری آنها را بررسی کردیم. در ادامه یک نمونهی کاملتر را مشاهده میکنید:
- در ابتدا نوعهای متداولی مانند number و string ذکر شدهاند.
- سپس نحوهی تعریف آرایهای از رشتهها را مشاهده میکنید.
- یا میتوان مقدار یک خاصیت را تنها به مقادیری خاص محدود کرد؛ مانند خاصیت status در اینجا و اگر در حین مقدار دهی این خاصیت، از مقدار دیگری استفاده شود، تایپاسکریپت، خطایی را صادر میکند.
- در ادامه سه روش تعریف اشیاء تو در تو را مشاهده میکنید؛ خاصیت thing از نوع یک شیء خالی تعریف شدهاست (بجای آن میتوان از نوع object هم استفاده کرد). خاصیت thing2 از نوع یک شیء که دارای خاصیت رشتهای name است، تعریف شده و یا بهتر است این نوع تعاریف را به یک type مستقل دیگر مانند User منتقل کرد و سپس از آن جهت تعیین نوع خاصیتی مانند user استفاده نمود.
- در اینجا حتی میتوان یک خاصیت را که از نوع یک تابع است، تعریف کرد. در این تعریف، void نوع خروجی آن است.
روش تعریف props تابعی در TypeScript
برای بررسی روش تعریف نوع توابع ارسالی از طریق props، ابتدا کامپوننت جدید src\components\Button.tsx را ایجاد میکنیم. سپس آنرا به صورت زیر تکمیل خواهیم کرد:
در این کامپوننت، متغیر onClick حاصل از Object Destructuring شیء props دریافتی، یک تابع است که قرار است با کلیک بر روی دکمهای که در این کامپوننت قرار دارد، پیامی را به کامپوننت والد ارسال کند.
با توجه به تعریف { onClick }، در همان لحظه، خطای any بودن نوع آن از طرف TypeScript گزارش داده میشود. بنابراین نوع جدید Props را ایجاد کرده و برای onClick، نوع متناسبی را تعریف میکنیم. در اینجا 4 روش مختلف تعریف نوع function را در TypeScript مشاهده میکنید؛ دو حالت آن با ذکر پرانتزها و درج امضای متد انجام شده و دو حالت دیگر به کمک arrow functions پیاده سازی شدهاند.
برای نمونه آخرین حالت تعریف شده از روش arrow functions استفاده میکند که متداولترین روش تعریف نوع توابع است (چون عنوان میکند که نوع onClick، یک تابع است و آنرا شبیه به یک متد معمولی نمایش نمیدهد) که در آن در ابتدا امضای پارامترهای این تابع مشخص شدهاند و در ادامه پس از <=، نوع خروجی این تابع تعریف شدهاست که void میباشد (این تابع چیزی را بر نمیگرداند).
در آخر، تعریف المان آنرا به صورت زیر به فایل src\App.tsx اضافه میکنیم که onClick آن یک مقدار را دریافت کرده و سپس آنرا در کنسول نمایش میدهد.
البته خروجی از نوع void، در اینجا بسیار معمول است؛ چون هدف از این نوع توابع بیشتر ارسال مقادیری به کامپوننت در برگیرندهی آنها است (مانند value در اینجا) و اگر برای مثال خروجی رشتهای را داشته باشند، باید در حین درج و تعریف المان آنها، برای نمونه یک "return "value1 را هم در انتهای کار قرار داد که عملا استفادهای ندارد و بیمعنا است:
روش تعیین props پیشفرض توسط TypeScript
اگر بخواهیم توسط روشهای خود React، مقادیر پیشفرض props را تعیین کنیم، میتوان از defaultProps به صورت زیر با تعریف یک شیء جاوا اسکریپتی از پیش مقدار دهی شده، استفاده کرد:
Head.defaultProps = { title: "Hello", isActive: true };
type Props = { title: string; isActive: boolean; }; export const Head = ({ title = "Hello", isActive = true }: Props) => {
در این حالت انتظار داریم که در حین استفاده و تعریف المان کامپوننت Head، اگر برای مثال ویژگی isActive را ذکر نکردیم، کامپایلر TypeScript خطایی را گزارش نکند؛ که اینطور نیست:
هنوز هم در اینجا میتوان خطای عدم تعریف خاصیت isActive را مشاهده کرد. برای رفع این مشکل، به صورت زیر عمل میکنیم:
type Props = { title: string; isActive?: boolean; }; export const Head = ({ title, isActive = true }: Props) => {
اکنون با تعریف isActive?: boolean، دیگر شاهد نمایش خطایی در حین تعریف المان Head، بدون ذکر خاصیت isActive، نخواهیم بود.
تعریف انواع و اقسام نوعهای props
تا اینجا نوعهای سادهای مانند string و boolean و همچنین نحوهی تعریف اجباری و اختیاری آنها را بررسی کردیم. در ادامه یک نمونهی کاملتر را مشاهده میکنید:
type User = { name: string; }; type Props = { title: string; isActive?: boolean; count: number; options: string[]; status: "loading" | "loaded"; thing: {}; thing2: { name: string; }; user: User; func: () => void; };
- سپس نحوهی تعریف آرایهای از رشتهها را مشاهده میکنید.
- یا میتوان مقدار یک خاصیت را تنها به مقادیری خاص محدود کرد؛ مانند خاصیت status در اینجا و اگر در حین مقدار دهی این خاصیت، از مقدار دیگری استفاده شود، تایپاسکریپت، خطایی را صادر میکند.
- در ادامه سه روش تعریف اشیاء تو در تو را مشاهده میکنید؛ خاصیت thing از نوع یک شیء خالی تعریف شدهاست (بجای آن میتوان از نوع object هم استفاده کرد). خاصیت thing2 از نوع یک شیء که دارای خاصیت رشتهای name است، تعریف شده و یا بهتر است این نوع تعاریف را به یک type مستقل دیگر مانند User منتقل کرد و سپس از آن جهت تعیین نوع خاصیتی مانند user استفاده نمود.
- در اینجا حتی میتوان یک خاصیت را که از نوع یک تابع است، تعریف کرد. در این تعریف، void نوع خروجی آن است.
روش تعریف props تابعی در TypeScript
برای بررسی روش تعریف نوع توابع ارسالی از طریق props، ابتدا کامپوننت جدید src\components\Button.tsx را ایجاد میکنیم. سپس آنرا به صورت زیر تکمیل خواهیم کرد:
import React from "react"; type Props = { // onClick(): string; method returns string // onClick(): void method returns nothing; // onClick(text: string): void; method with params // onClick: () => void; function returns nothing onClick: (text: string) => void; // function with params }; export const Button = ({ onClick }: Props) => { return <button onClick={() => onClick("hi")}>Click Me</button>; };
با توجه به تعریف { onClick }، در همان لحظه، خطای any بودن نوع آن از طرف TypeScript گزارش داده میشود. بنابراین نوع جدید Props را ایجاد کرده و برای onClick، نوع متناسبی را تعریف میکنیم. در اینجا 4 روش مختلف تعریف نوع function را در TypeScript مشاهده میکنید؛ دو حالت آن با ذکر پرانتزها و درج امضای متد انجام شده و دو حالت دیگر به کمک arrow functions پیاده سازی شدهاند.
برای نمونه آخرین حالت تعریف شده از روش arrow functions استفاده میکند که متداولترین روش تعریف نوع توابع است (چون عنوان میکند که نوع onClick، یک تابع است و آنرا شبیه به یک متد معمولی نمایش نمیدهد) که در آن در ابتدا امضای پارامترهای این تابع مشخص شدهاند و در ادامه پس از <=، نوع خروجی این تابع تعریف شدهاست که void میباشد (این تابع چیزی را بر نمیگرداند).
در آخر، تعریف المان آنرا به صورت زیر به فایل src\App.tsx اضافه میکنیم که onClick آن یک مقدار را دریافت کرده و سپس آنرا در کنسول نمایش میدهد.
البته خروجی از نوع void، در اینجا بسیار معمول است؛ چون هدف از این نوع توابع بیشتر ارسال مقادیری به کامپوننت در برگیرندهی آنها است (مانند value در اینجا) و اگر برای مثال خروجی رشتهای را داشته باشند، باید در حین درج و تعریف المان آنها، برای نمونه یک "return "value1 را هم در انتهای کار قرار داد که عملا استفادهای ندارد و بیمعنا است:
import { Button } from "./components/Button"; import { Head } from "./components/Head"; // ... function App() { return ( <div className="App"> <Head title="Hello" /> <Button onClick={(value) => { console.log(value); }} /> // ...