در گریدی که تا به اینجا طراحی کردیم، اگر قرار باشد بجای جدول فیلمها، جدول مشتریها نمایش داده شود، چکار باید کرد؟ با پیاده سازی فعلی، باید کل تعاریف MoviesTable را در کامپوننت دیگری مانند CustomersTable تکرار کنیم. به همین جهت برای پویاسازی تعاریف ستونها نیاز است این قسمت را از جدول اصلی جدا کرده و به کامپوننت مستقلی مانند tableHeader منتقل کنیم.
ایجاد کامپوننت جدید tableHeader
برای پویاسازی تعاریف ستونها و همچنین کم کردن مسئولیتهای کامپوننت MoviesTable، فایل جدید src\components\common\tableHeader.jsx را ایجاد میکنیم تا در برگیرندهی کامپوننت جدید TableHeader شود. پس از ایجاد این فایل، با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت TableHeader را تشکیل میدهیم. سپس به کامپوننت MoviesTable بازگشته و متد raiseSort آنرا cut و به اینجا منتقل میکنیم. همچنین نیاز است کل thead جدول فیلمها را نیز به اینجا منتقل کنیم. اما چون میخواهیم این تعاریف پویا باشند، باید امکان تعریف پویای ستونها را نیز به آن اضافه کنیم. بنابراین اینترفیس این کامپوننت به صورت زیر است:
- ورودیهای آن: آرایهی ستونهای جدول و همچنین شیء sortColumn و رخداد onSort که در متد raiseSort استفاده میشوند.
با این توضیحات، کامپوننت TableHeader چنین شکلی را پیدا میکند:
در ابتدای آن، متد raiseSort را از کامپوننت MoviesTable به اینجا منتقل کردهایم.
سپس در متد رندر آن، بر اساس آرایهی columns که از props این کامپوننت دریافت خواهد شد، لیست thهای هدر را به صورت پویا رندر میکنیم. در اینجا ساختار مورد نیاز شیء column را نیز مشاهده میکنید. نیاز است یک برچسب نمایش داده شود و همچنین برای اینکه this.raiseSort نیز بتواند مجددا کار کند، نیاز است نام خاصیتی که قرار است مرتب سازی بر اساس آن انجام شود نیز مشخص باشد. بنابراین تا اینجا شیء column باید دارای دو خاصیت label و path باشد.
پس از تعریف ابتدایی کامپوننت TableHeader، به کامپوننت MoviesTable بازگشته و شروع به استفادهی از آن میکنیم:
در ادامه باید آرایهی columns را که به صورت props به کامپوننت TableHeader ارسال میشود، تعریف و مقدار دهی کنیم که تشکیل شدهاست از اشیایی با خواص path و label:
در اینجا دو شیء خالی را نیز در انتهای لیست مشاهده میکنید که به thهای خالی مانند نمایش Like و دکمهی Delete اشاره میکنند.
اکنون میتوان کل تعریف thead موجود در این کامپوننت را به طور کامل با کامپوننت TableHeader ای که import کردیم، جایگزین کنیم:
در اینجا ویژگیهای مورد نیاز جهت تامین props کامپوننت TableHeader نیز ذکر شدهاند. this.columns را که در همین کامپوننت تعریف کردیم، sortColumn و onSort هم جزو props ارسالی به کامپوننت جاری هستند.
در این حالت اگر برنامه را اجرا کنید، بدون مشکل خروجی نهایی را رندر میکند؛ اما در کنسول توسعه دهندگان مرورگر یک چنین خطایی را نیز لاگ خواهد کرد:
در حین تعریف رندر لیست thها در کامپوننت TableHeader، ذکر ویژگی key را فراموش کردهایم. البته در اینجا میتوان از column.path بهعنوان key استفاده کرد، اما چون در آرایهی ستونها دو شیء خالی را نیز در انتهای لیست داریم، بهتر است برای اینها یک id را نیز تعریف کردیم تا بتوان آنها را به صورت منحصربفردی شناسایی کرد:
سپس متد رندر کامپوننت TableHeader را جهت درج key به روز رسانی میکنیم:
دراینجا اگر column.path مقدار دهی شده بود، از آن استفاده میشود، در غیراینصورت از مقدار column.key، به عنوان مقدار ویژگی خاصیت key هر المان th، استفاده خواهد شد.
استخراج TableBody از جدول کامپوننت MoviesTable
اکنون با استخراج TableHeader از کامپوننت MoviesTable، به همان مشکل مخلوط بودن درجهی abstractions رسیدهایم. از یک طرف با یک abstraction سطح بالا مانند TableHeader در این کامپوننت سر و کار داریم و از طرف دیگر، نمایش تمام جزئیات درونی رندر جدول نیز پیش روی ما است. همچنین رندر ستونهای آن نیز پویا نیست و هنوز بر اساس خاصیت this.columns تعریف شده، واکنش نشان نمیدهد. به همین جهت tbody این جدول را نیز به یک کامپوننت مستقل تبدیل میکنیم. برای این منظور فایل جدید src\components\common\tableBody.jsx را اضافه میکنیم. سپس با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت TableBody را تشکیل میدهیم.
این کامپوننت قرار است آرایهای از اشیاء را دریافت و ردیفهایی را بر اساس آنها رندر کند. به همین جهت این آرایه را از props و با نام data دریافت میکنیم. نام data به عمد انتخاب شدهاست، تا بیانگر عمومی بودن آن باشد؛ بجای استفاده از نام ویژهی آرایهی movies، در این مثال خاص.
تا اینجا ساختار ابتدایی کامپوننت TableBody را مشاهده میکنید که هدف آن، رندر پویای قسمت tbody جدول است. این کامپوننت ابتدا نیاز دارد تا data را از props دریافت کند و بر اساس آن، لیست trها را رندر کند. سپس هر tr نیز از چندین td تشکیل میشود. به همین جهت به لیست دومی به نام columns، برای رندر پویای tdها نیاز است.
رندر محتویات هر سلول جدول به صورت پویا
در این مرحله میخواهیم محتویات tdها را رندر کنیم و حالت فعلی آنها یک چنین شکلی را داشته و در آن ارجاع مستقیمی به شیء movie و خواص آن وجود دارد:
به علاوه این tdها به رندر دکمهی Like و Delete که المانهای سفارشی نیز محسوب میشوند، ختم شدهاند.
برای رندر خواص اشیاء آرایهی ارسالی به کامپوننت TableBody، میتوان از روش [] برای دسترسی به مقادیر خواص استفاده کرد که سبب رندر پویای این مقادیر میشود:
مشکل! روش item[column.path] با خاصیتی مانند "genre.name" که یک خاصیت تو در تو است، کار نمیکند. به همین جهت نیاز به متد زیر، برای انجام اینکار است:
بنابراین تا اینجا روش رندر مقدار هر خاصیت به صورت زیر تغییر میکند:
این تغییر میتواند 4 ستون اول را بدون مشکل رندر کند. اما برای مثال در ستون پنجم، کامپوننت Like قرار گرفتهاست. برای نمایش آن باید چکار کرد؟
همانطور که در ابتدای این سری نیز بررسی کردیم، عبارات JSX در نهایت به اشیاء خالص جاوا اسکریپتی ترجمه میشوند. این ویژگی در حین تعریف المانهای سفارشی مانند کامپوننت Like نیز صادق است. به همین جهت در آرایهی columns که تعاریف ستونهای جدول را به همراه دارد، میتوان یک خاصیت جدید را تعریف و به آن عبارات JSX را انتساب داد. بنابراین تعاریف tdهای Like و Delete را به طور کامل cut کرده و به خاصیت جدید content این دو شیء خالی انتهای لیست آرایهی columns انتساب میدهیم:
البته در اینجا جهت مقدار دهی اشیایی مانند movie، بجای استفادهی مستقیم از یک React element، از یک arrow function استفاده کردهایم تا movie را دریافت کند و یک المان React را بازگشت دهد. همچنین پیشتر از متغیرهای onLike و onDelete در کدهای tdها استفاده کرده بودیم که در ابتدای متد رندر تعریف شده بودند؛ اما زمانیکه این قطعات کد را به خاصیت content منتقل میکنیم، دیگر شناسایی نمیشوند. بنابراین در اینجا برای دسترسی به آنها، مستقیما از props استفاده میشود.
مرحلهی بعد، مراجعه به کامپوننت tableBody و استفاده از خاصیت جدید content، جهت رندر محتوای آن است. در اینجا در متد renderCell بررسی میکنیم اگر ستونی دارای خاصیت content باشد، آن content را رندر میکنیم. در غیراینصورت از همان getPropValue متداول استفاده خواهد شد:
- در متد renderCell، فراخوانی column.content(item) با توجه به function بودن content تعریف شدهی در آرایهی columns، در حقیقیت یک عبارت JSX را بازگشت میدهد که در خروجیهای متدهای React مجاز است و در نهایت تبدیل به المانهای خالص جاوا اسکریپتی در DOM مجازی React و در نهایت DOM اصلی مرورگر میشوند.
- همچنین در اینجا یک createKey را نیز مشاهده میکنید. المانهای هر Array.map نوشته شده، نیاز به یک ویژگی key مقدار دهی شده دارند که در دو قسمت trها و همچنین tdها تعریف شدهاست. در فرمول آن جائیکه از || استفاده شده، اگر ستونی دارای path بود، مقدار آن درج میشود، اما اگر مانند دو ستون آخر صرفا key تعریف شده بود، وجود || سبب میشود تا column.key درنظر گرفته شود و مشکلی رخ ندهد.
- علت تعریف دو متد مجزای renderCell و createKey هم کم شدن بار if/elseها، در بین کدهای درج شدهی در ردیفهای جدول است.
اکنون به کامپوننت MoviesTable مراجعه کرده و کل tbody آنرا حذف و با المان کامپوننت TableBody، جایگزین میکنیم:
تا اینجا اگر این تغییرات را ذخیره کرده و برنامه را مجددا در مرورگر بارگذاری کنیم، باید به همان خروجی قبلی برسیم؛ که اینبار تعاریف ستونهای آن پویا شدهاست.
اضافه کردن آیکن مرتب سازی اطلاعات به سر ستونهای جدول
در کامپوننت tableHeader، کار رندر thها انجام میشود. در اینجا پس از نام سرستون، میخواهیم آیکن نمایش صعودی و یا نزولی بودن روش مرتب سازی جاری را نمایش دهیم. برای این منظور، ابتدا متد renderSortIcon را به این کامپوننت اضافه میکنیم:
این متد، شیء column در حال رندر را دریافت کرده و بر اساس sortColumn دریافتی از props و همچنین صعودی و یا نزولی بودن روش مرتب سازی، یکی از آیکنهای font-awesome را به صورت یک المان جدید رندر میکند. اگر این column در حال رندر، با sortColumn تعیین شده یکی نبود، آیکنی رندر نمیشود (با بازگشت نال، هیچ چیزی رندر نخواهد شد).
و سپس در متد رندر کامپوننت tableHeader، این متد را در کنار label آن ستون درج خواهیم کرد:
پس از ذخیره سازی تغییرات و بارگذاری مجدد برنامه در مرورگر، خروجی آنرا برای نمونه به صورت یک آیکن مثلثی شکل، در کنار عنوان Title میتوان مشاهده کرد:
استخراج کل Table از جدول کامپوننت MoviesTable
در حال حاضر اگر به پیاده سازی کامپوننت MoviesTable دقت کنیم، یک تگ table به همراه دو کامپوننت TableHeader و TableBody در آن درج شدهاند. با این طراحی، اگر قصد استفادهی از این امکانات را در جای دیگری داشته باشیم، باید دقیقا همین قطعه کد را تکرار کنیم. به همین جهت کل تگ table این کامپوننت را استخراج کرده و به کامپوننت جدیدی منتقل میکنیم. به همین جهت فایل جدید src\components\common\table.jsx را ایجاد کرده و با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت Table را تشکیل میدهیم. سپس کل تگ table کامپوننت MoviesTable را cut کرده و به متد رندر کامپوننت جدید Table منتقل میکنیم. سپس اولین قدم برای سازگار کردن این محتوا با یک کامپوننت جدید، افزودن importهای زیر است:
سپس باید تمام ویژگیهای استفاده شدهی در این المان منتقل شده را از طریق props دریافت کرد که انجام اینکار را در سطر اول متد رندر مشاهده میکنید:
با این تغییرات به یک کامپوننت سادهی با قابلیت استفادهی مجدد رسیدهایم. اکنون المان آنرا در کامپوننت MoviesTable، در جای تگ قبلی table قرار میدهیم:
کدهای کامل این قسمت را از اینجا میتوانید دریافت کنید: sample-14.zip
ایجاد کامپوننت جدید tableHeader
برای پویاسازی تعاریف ستونها و همچنین کم کردن مسئولیتهای کامپوننت MoviesTable، فایل جدید src\components\common\tableHeader.jsx را ایجاد میکنیم تا در برگیرندهی کامپوننت جدید TableHeader شود. پس از ایجاد این فایل، با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت TableHeader را تشکیل میدهیم. سپس به کامپوننت MoviesTable بازگشته و متد raiseSort آنرا cut و به اینجا منتقل میکنیم. همچنین نیاز است کل thead جدول فیلمها را نیز به اینجا منتقل کنیم. اما چون میخواهیم این تعاریف پویا باشند، باید امکان تعریف پویای ستونها را نیز به آن اضافه کنیم. بنابراین اینترفیس این کامپوننت به صورت زیر است:
- ورودیهای آن: آرایهی ستونهای جدول و همچنین شیء sortColumn و رخداد onSort که در متد raiseSort استفاده میشوند.
با این توضیحات، کامپوننت TableHeader چنین شکلی را پیدا میکند:
import React, { Component } from "react"; class TableHeader extends Component { raiseSort = path => { console.log("raiseSort", path); const sortColumn = { ...this.props.sortColumn }; if (sortColumn.path === path) { sortColumn.order = sortColumn.order === "asc" ? "desc" : "asc"; } else { sortColumn.path = path; sortColumn.order = "asc"; } this.props.onSort(sortColumn); }; render() { return ( <thead> <tr> {this.props.columns.map(column => ( <th onClick={() => this.raiseSort(column.path)}>{column.label}</th> ))} </tr> </thead> ); } } export default TableHeader;
سپس در متد رندر آن، بر اساس آرایهی columns که از props این کامپوننت دریافت خواهد شد، لیست thهای هدر را به صورت پویا رندر میکنیم. در اینجا ساختار مورد نیاز شیء column را نیز مشاهده میکنید. نیاز است یک برچسب نمایش داده شود و همچنین برای اینکه this.raiseSort نیز بتواند مجددا کار کند، نیاز است نام خاصیتی که قرار است مرتب سازی بر اساس آن انجام شود نیز مشخص باشد. بنابراین تا اینجا شیء column باید دارای دو خاصیت label و path باشد.
پس از تعریف ابتدایی کامپوننت TableHeader، به کامپوننت MoviesTable بازگشته و شروع به استفادهی از آن میکنیم:
import TableHeader from "./common/tableHeader";
در ادامه باید آرایهی columns را که به صورت props به کامپوننت TableHeader ارسال میشود، تعریف و مقدار دهی کنیم که تشکیل شدهاست از اشیایی با خواص path و label:
columns = [ { path: "title", label: "Title" }, { path: "genre.name", label: "Genre" }, { path: "numberInStock", label: "Stock" }, { path: "dailyRentalRate", label: "Rate" }, {}, {} ];
اکنون میتوان کل تعریف thead موجود در این کامپوننت را به طور کامل با کامپوننت TableHeader ای که import کردیم، جایگزین کنیم:
render() { const { movies, onDelete, onLike, onSort, sortColumn } = this.props; return ( <table className="table"> <TableHeader columns={this.columns} sortColumn={sortColumn} onSort={onSort} /> <tbody>
در این حالت اگر برنامه را اجرا کنید، بدون مشکل خروجی نهایی را رندر میکند؛ اما در کنسول توسعه دهندگان مرورگر یک چنین خطایی را نیز لاگ خواهد کرد:
index.js:1375 Warning: Each child in a list should have a unique "key" prop. Check the render method of `TableHeader`. See https://fb.me/react-warning-keys for more information.
class MoviesTable extends Component { columns = [ { path: "title", label: "Title" }, { path: "genre.name", label: "Genre" }, { path: "numberInStock", label: "Stock" }, { path: "dailyRentalRate", label: "Rate" }, { key: "like" }, { key: "delete" } ];
render() { return ( <thead> <tr> {this.props.columns.map(column => ( <th key={column.path || column.key} style={{ cursor: "pointer" }} onClick={() => this.raiseSort(column.path)} > {column.label} </th> ))} </tr> </thead> );
استخراج TableBody از جدول کامپوننت MoviesTable
اکنون با استخراج TableHeader از کامپوننت MoviesTable، به همان مشکل مخلوط بودن درجهی abstractions رسیدهایم. از یک طرف با یک abstraction سطح بالا مانند TableHeader در این کامپوننت سر و کار داریم و از طرف دیگر، نمایش تمام جزئیات درونی رندر جدول نیز پیش روی ما است. همچنین رندر ستونهای آن نیز پویا نیست و هنوز بر اساس خاصیت this.columns تعریف شده، واکنش نشان نمیدهد. به همین جهت tbody این جدول را نیز به یک کامپوننت مستقل تبدیل میکنیم. برای این منظور فایل جدید src\components\common\tableBody.jsx را اضافه میکنیم. سپس با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت TableBody را تشکیل میدهیم.
این کامپوننت قرار است آرایهای از اشیاء را دریافت و ردیفهایی را بر اساس آنها رندر کند. به همین جهت این آرایه را از props و با نام data دریافت میکنیم. نام data به عمد انتخاب شدهاست، تا بیانگر عمومی بودن آن باشد؛ بجای استفاده از نام ویژهی آرایهی movies، در این مثال خاص.
import React, { Component } from "react"; class TableBody extends Component { render() { const { data, columns } = this.props; return ( <tbody> {data.map(item => ( <tr> {columns.map(column => ( <td></td> ))} </tr> ))} </tbody> ); } } export default TableBody;
رندر محتویات هر سلول جدول به صورت پویا
در این مرحله میخواهیم محتویات tdها را رندر کنیم و حالت فعلی آنها یک چنین شکلی را داشته و در آن ارجاع مستقیمی به شیء movie و خواص آن وجود دارد:
{movies.map(movie => ( <tr key={movie._id}> <td>{movie.title}</td>
برای رندر خواص اشیاء آرایهی ارسالی به کامپوننت TableBody، میتوان از روش [] برای دسترسی به مقادیر خواص استفاده کرد که سبب رندر پویای این مقادیر میشود:
<td>{item[column.path]}</td>
getPropValue(obj, path) { if (!path) { return obj; } const properties = path.split("."); return this.getPropValue(obj[properties.shift()], properties.join(".")); }
<td>{getPropValue(item, column.path)}</td>
همانطور که در ابتدای این سری نیز بررسی کردیم، عبارات JSX در نهایت به اشیاء خالص جاوا اسکریپتی ترجمه میشوند. این ویژگی در حین تعریف المانهای سفارشی مانند کامپوننت Like نیز صادق است. به همین جهت در آرایهی columns که تعاریف ستونهای جدول را به همراه دارد، میتوان یک خاصیت جدید را تعریف و به آن عبارات JSX را انتساب داد. بنابراین تعاریف tdهای Like و Delete را به طور کامل cut کرده و به خاصیت جدید content این دو شیء خالی انتهای لیست آرایهی columns انتساب میدهیم:
class MoviesTable extends Component { columns = [ { path: "title", label: "Title" }, { path: "genre.name", label: "Genre" }, { path: "numberInStock", label: "Stock" }, { path: "dailyRentalRate", label: "Rate" }, { key: "like", content: movie => ( <Like liked={movie.liked} onClick={() => this.props.onLike(movie)} /> ) }, { key: "delete", content: movie => ( <button onClick={() => this.props.onDelete(movie)} className="btn btn-danger btn-sm" > Delete </button> ) } ];
مرحلهی بعد، مراجعه به کامپوننت tableBody و استفاده از خاصیت جدید content، جهت رندر محتوای آن است. در اینجا در متد renderCell بررسی میکنیم اگر ستونی دارای خاصیت content باشد، آن content را رندر میکنیم. در غیراینصورت از همان getPropValue متداول استفاده خواهد شد:
renderCell = (item, column) => { if (column.content) { return column.content(item); } return this.getPropValue(item, column.path); }; createKey = (item, column) => { return item._id + (column.path || column.key); }; render() { const { data, columns } = this.props; return ( <tbody> {data.map(item => ( <tr key={item._id}> {columns.map(column => ( <td key={this.createKey(item, column)}> {this.renderCell(item, column)} </td> ))} </tr> ))} </tbody> ); }
- همچنین در اینجا یک createKey را نیز مشاهده میکنید. المانهای هر Array.map نوشته شده، نیاز به یک ویژگی key مقدار دهی شده دارند که در دو قسمت trها و همچنین tdها تعریف شدهاست. در فرمول آن جائیکه از || استفاده شده، اگر ستونی دارای path بود، مقدار آن درج میشود، اما اگر مانند دو ستون آخر صرفا key تعریف شده بود، وجود || سبب میشود تا column.key درنظر گرفته شود و مشکلی رخ ندهد.
- علت تعریف دو متد مجزای renderCell و createKey هم کم شدن بار if/elseها، در بین کدهای درج شدهی در ردیفهای جدول است.
اکنون به کامپوننت MoviesTable مراجعه کرده و کل tbody آنرا حذف و با المان کامپوننت TableBody، جایگزین میکنیم:
//... import TableBody from "./common/tableBody"; //... class MoviesTable extends Component { // ... render() { const { movies, onSort, sortColumn } = this.props; return ( <table className="table"> <TableHeader columns={this.columns} sortColumn={sortColumn} onSort={onSort} /> <TableBody columns={this.columns} data={movies} /> </table> ); } }
اضافه کردن آیکن مرتب سازی اطلاعات به سر ستونهای جدول
در کامپوننت tableHeader، کار رندر thها انجام میشود. در اینجا پس از نام سرستون، میخواهیم آیکن نمایش صعودی و یا نزولی بودن روش مرتب سازی جاری را نمایش دهیم. برای این منظور، ابتدا متد renderSortIcon را به این کامپوننت اضافه میکنیم:
renderSortIcon = column => { const { sortColumn } = this.props; if (column.path !== sortColumn.path) { return null; } if (sortColumn.order === "asc") { return <i className="fa fa-sort-asc" />; } return <i className="fa fa-sort-desc" />; };
و سپس در متد رندر کامپوننت tableHeader، این متد را در کنار label آن ستون درج خواهیم کرد:
{column.label} {this.renderSortIcon(column)}
استخراج کل Table از جدول کامپوننت MoviesTable
در حال حاضر اگر به پیاده سازی کامپوننت MoviesTable دقت کنیم، یک تگ table به همراه دو کامپوننت TableHeader و TableBody در آن درج شدهاند. با این طراحی، اگر قصد استفادهی از این امکانات را در جای دیگری داشته باشیم، باید دقیقا همین قطعه کد را تکرار کنیم. به همین جهت کل تگ table این کامپوننت را استخراج کرده و به کامپوننت جدیدی منتقل میکنیم. به همین جهت فایل جدید src\components\common\table.jsx را ایجاد کرده و با استفاده از میانبرهای imrc و cc، ساختار ابتدایی کامپوننت Table را تشکیل میدهیم. سپس کل تگ table کامپوننت MoviesTable را cut کرده و به متد رندر کامپوننت جدید Table منتقل میکنیم. سپس اولین قدم برای سازگار کردن این محتوا با یک کامپوننت جدید، افزودن importهای زیر است:
import TableBody from "./tableBody"; import TableHeader from "./tableHeader";
import TableBody from "./tableBody"; import TableHeader from "./tableHeader"; class Table extends Component { render() { const { columns, sortColumn, onSort, data } = this.props; return ( <table className="table"> <TableHeader columns={columns} sortColumn={sortColumn} onSort={onSort} /> <TableBody columns={columns} data={data} /> </table> ); } } export default Table;
//... import Table from "./common/table"; class MoviesTable extends Component { //... render() { const { movies, onSort, sortColumn } = this.props; return ( <Table columns={this.columns} sortColumn={sortColumn} onSort={onSort} data={movies} /> ); } }
کدهای کامل این قسمت را از اینجا میتوانید دریافت کنید: sample-14.zip