اهمیت Breadcrumbها
در گذشته، توسعه دهندگان وب بیشتر نگران توابع و اطلاعاتی که وباپلیکیشنها میتوانستند تحویل دهند (backend) بودند، و زیاد به ظاهر بصری (frontend) صفحات وب اهمیت نمیدادند. این مسئله تقصیر آنها نبود. در آن زمان فناوری مناسبی برای توسعه صفحات وب پیچیده وجود نداشت. موارد کمی هم که میتوانستند این کار را انجام دهند، استفاده سختی داشتند.
امروزه این مسئله یک داستان کاملا متفاوت است؛ جایگاه طراحی رابط کاربری / تجربه کاربری در توسعه وب، درست به اندازه برنامهریزی منطق backend که همه چیز را در حالت اجرا نگه میدارد، مهم است. طیف بزرگی از الگوهای طراحی قابل قبول وجود دارد که میتواند به یک وباپلیکیشن کمک کند تا برچسب «کاربر دوست» را دریافت کند، اما حتی اشتباهات طراحی بیشتری وجود دارند که همه باید مواظب آنها باشند.
مردم از این که صفحات popup ناخواسته را بر روی صفحه ببینند، متنفرند. من همینطور هستم، اما popupها تنها مواردی نیستند که میتوانند جهت راه کاربر را تغییر دهند.
این مسئله میتواند یک مشکل جدی باشد، اما به هیچ وجه اجباری نیست. Breadcrumbها میتوانند سادهترین راه حل برای یک تجربه کاربری نرم برای کاربر باشند.
Breadcrumbها یک سیستم جهتیابی فراهم میکنند تا به کاربران در دانستن موقعیت فعلی خود در رابطه با دیگر صفحات بر روی یک وبسایت کمک کنند. نام Breadcrumb از داستان معروف، یعنی Hansel and Grettel گرفته شده است، زیرا تقریبا همان اتفاق میافتد. یک رد باقی گذاشته میشود تا از گم شدن جلوگیری شود، و در نتیجه تجربه کاربر ترقی داده شود. Breadcrumbها بر روی یک وبسایت، همچنین تعداد حرکاتی که یک کاربر باید برای رسیدن به یک صفحه انجام دهد را کاهش میدهند.
راهاندازی Breadcrumbها در Laravel بسیار ساده است. پکیجی وجود دارد که به اکثر منطق این قضیه رسیدگی میکند، و ما به نحوه استفاده از این پکیج و رسیدن به بهترین نتیجه نگاهی خواهیم داشت.
راهاندازی برنامه Laravel
کلیت این مقاله، توسعه دهندگان Laravel را مورد هدف قرار میدهد.
ما پکیج Laravel Breadcrumbs را از طریق Composer وارد خواهیم کرد، و کد مورد نظر را برای رندر کردن جهتیابی سرویس Breadcrumb به صورت دینامیک، بر حسب صفحهای که کاربر در حال مرور است خواهیم نوشت.
کد نهایی این مقاله، در این صفحه بر روی گیتهاب در دسترس است.
در جهت سادهتر شدن کار، ما یک نمونه برنامه Laravel جدید را نصب خواهیم کرد.
laravel new breadcrumbs
یا
composer create-project --prefer-dist laravel/laravel breadcrumbs
سپس با نوشتن دستور زیر در ترمینال، پکیج Laravel Breadcrumbs را وارد میکنیم:
composer require davejamesmiller/laravel-breadcrumbs
ساخت HTTP routing
بیایید برخی routeهای HTML را در فایل routes/web.php بسازیم. (و همچنین نامگذاری کنیم) ما در بخشهای بعدی، با استفاده از نام این routeها به آنها اشاره خواهیم کرد.
اینها routeهایی هستند که ما برای این مثال نیاز خواهیم داشت:
routes/web.php:
Route::get('/', ['as' => 'home', 'uses' => 'MainController@home']);
Route::get('/continent/{name}', ['as' => 'continent', 'uses' => 'MainController@continent']);
Route::get('/country/{name}', ['as' => 'country', 'uses' => 'MainController@country']);
Route::get('/city/{name}', ['as' => 'city', 'uses' => 'MainController@city']);
در جهت نشان دادن قدرت Laravel Breadcrumbs، ما یک برنامه کوچک خواهیم ساخت که در آن مدلهای قاره، کشور و شهر را ثبت میکنیم. ما همچنین خواهیم گفت که یک قاره چند کشور دارد (hasMany) و یک کشور چند شهر دارد. (hasMany)
برای ساخت این مدلها، این دستورات را در ترمینال مینویسیم:
php artisan make:model Continent
php artisan make:model Country
php artisan make:model City
همچنین بیایید فایل مهاجرت را برای هر کدام از این مدلها بسازیم:
php artisan make:migration continents
php artisan make:migration countries
php artisan make:migration cities
سپس، بیایید متدهای رابطه (relationship) را در هر کدام از این کلاسهای مدل تعریف کنیم:
app/Continent.php:
namespace App;
use App\Country;
use Illuminate\Database\Eloquent\Model;
class Continent extends Model
{
public function country(){
return $this->hasMany(Country::class);
}
}
app/Country.php:
namespace App;
use App\City;
use App\Continent;
use Illuminate\Database\Eloquent\Model;
class Country extends Model
{
protected $guarded = [];
public function city(){
return $this->hasMany(City::class);
}
public function continent(){
return $this->belongsTo(Continent::class);
}
}
app/City.php:
namespace App;
use App\Country;
use Illuminate\Database\Eloquent\Model;
class City extends Model
{
protected $guarded = [];
public function country(){
return $this->belongsTo(Country::class);
}
}
دقت کنید که guarded برابر با یک آرایه خالی قرار داده شده است؛ زیرا ما میخواهیم وقتی که در حال نوشتن بر روی دیتابیس هستیم، نوعی اختصاص دهی بزرگ انجام دهیم. هر زمان که یک تغییر این چنینی به یک فایل مدل اعمال میکنید، همیشه به یاد داشته باشید که قبل از رسیدن برنامه به مرحله تولید، این تغییر را برگردانید.
حال بیایید یک کنترلر بسازیم که درخواستهای HTTP که برنامه دریافت میکند را مدیریت خواهد کرد. ما این کنترلر را MainController نامگذاری کرده، و actionهای (methodهای) مختلفی به نمونههای compact مدل با view برگشتی تعریف خواهیم کرد. هر view، Breadcrumb(های) مخصوص خود را نمایش خواهد داد.
برای ساخت این کنترلر، این دستور را در ترمینال مینویسیم:
php artisan make:controller MainController
حال بیایید actionها (methodها)ای که viewها را پس از مدیریت درخواستهای HTTP بر میگردانند، بنویسیم.
app/Http/Controllers/MainController.php:
namespace App\Http\Controllers;
use App\Continent;
use App\Country;
use App\City;
use Illuminate\Http\Request;
class MainController extends Controller
{
public function home(){
return view('home');
}
public function continent($name){
$continent = Continent::where('name', $name)->first();
return view('continent', compact('continent'));
}
public function country($name){
$country = Country::where('name', $name)->first();
return view('country', compact('country'));
}
public function city($name){
$city = City::where('name', $name)->first();
return view('city', compact('city'));
}
}
ما رابطههای میان مدلها را include کردهایم، تا بتوانیم قابلیتهای پکیج Laravel Breadcrumbs را وقتی که به لینک کردن دینامیک و ویژگیهای ارتباطی میرسد، ببینیم.
حال بیایید برخی viewها را بسازیم، که با مدلهایی که ساختیم برابری خواهد داشت. در شاخه resources/views، چهار فایل جدید اضافه میکنیم. این ۴ فایل را به این صورت نامگذاری میکنیم:
home.blade.php
continent.blade.php
country.blade.php
city.blade.php
نامهای آنها کاملا خلاقانه هستند. اولین مورد کد frontend را برای صفحه خانه نگه میدارد، دومین مورد کد frontend را برای صفحه قارهها نگه میدارد، سومین مورد کد frontend را برای صفحه کشورها نگه میدارد و چهارمین مورد کد frontend را برای صفحه شهرها نگه میدارد.
خب، حال بیایید مقداری کد در این viewهای جدید قرار دهیم.
resources/views/home.blade.php:
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/css/bootstrap.min.css" integrity="sha384-rwoIResjU2yc3z8GV/NPeZWAv56rSmLldC3R/AZzGRnGxQQKnKkoFVhFQhNUwEyJ" crossorigin="anonymous">
</head>
<body>
{{ Breadcrumbs::render('home') }}
</body>
</html>
resources/views/continent.blade.php:
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/css/bootstrap.min.css" integrity="sha384-rwoIResjU2yc3z8GV/NPeZWAv56rSmLldC3R/AZzGRnGxQQKnKkoFVhFQhNUwEyJ" crossorigin="anonymous">
</head>
<body>
{{ Breadcrumbs::render('continent', $continent) }}
</body>
</html>
resources/views/country.blade.php:
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/css/bootstrap.min.css" integrity="sha384-rwoIResjU2yc3z8GV/NPeZWAv56rSmLldC3R/AZzGRnGxQQKnKkoFVhFQhNUwEyJ" crossorigin="anonymous">
</head>
<body>
{{ Breadcrumbs::render('country', $country->continent, $country) }}
</body>
</html>
resources/views/city.blade.php:
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0-alpha.6/css/bootstrap.min.css" integrity="sha384-rwoIResjU2yc3z8GV/NPeZWAv56rSmLldC3R/AZzGRnGxQQKnKkoFVhFQhNUwEyJ" crossorigin="anonymous">
</head>
<body>
{{ Breadcrumbs::render('city', $city->country->continent, $city->country, $city) }}
</body>
</html>
مدلهای Continent و Country نمایانگر یک رابطه «یک به چند» هستند. تعداد زیای شهر به یک کشور تعلق دارند، و تعداد زیادی کشور به یک قاره تعلق دارند. قاره، پدربزرگ شهر است و کشور والد آن است. ما به منطقی که رابطههای Laravel را تشکیل میدهد وارد نخواهیم شد.
در آخر، بیایید فایلهای مهاجرت را کمی تغییر دهیم تا ساختار صحیح برای جدول در دیتابیس را تعریف کنند.
database/migrations/2017_11_02_092826_continent.php:
use Illuminate\Support\Facades\Schema;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Database\Migrations\Migration;
class Continents extends Migration
{
/**
* مهاجرتها را اجرا کن.
*
* @return void
*/
public function up()
{
Schema::create('continents', function (Blueprint $table) {
$table->increments('id');
$table->string('name');
$table->timestamps();
});
}
/**
* مهاجرتها را برعکس کن..
*
* @return void
*/
public function down()
{
//
}
}
database/migrations/2017_11_02_092835_countries.php:
use Illuminate\Support\Facades\Schema;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Database\Migrations\Migration;
class Countries extends Migration
{
/**
* مهاجرتها را اجرا کن.
*
* @return void
*/
public function up()
{
Schema::create('countries', function (Blueprint $table) {
$table->increments('id');
$table->string('name');
$table->string('continent_id');
$table->timestamps();
});
}
/**
* مهاجرتها را برعکس کن.
*
* @return void
*/
public function down()
{
//
}
}
database/migrations/2017_11_02_092845_cities.php:
use Illuminate\Support\Facades\Schema;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Database\Migrations\Migration;
class Cities extends Migration
{
/**
* مهاجرتها را اجرا کن.
*
* @return void
*/
public function up()
{
Schema::create('cities', function (Blueprint $table) {
$table->increments('id');
$table->string('name');
$table->string('country_id');
$table->timestamps();
});
}
/**
* مهاجرتها را برعکس کن.
*
* @return void
*/
public function down()
{
//
}
}
ما میتوانیم مهاجرت را با استفاده از این دستور در ترمینال اجرا کنیم:
php artisan migrate
بیایید مقداری داده به دیتابیس وارد کنیم تا بتوانیم برنامه را امتحان کنیم. برای انجام این کار، سه فایل factory میسازیم (هر کدام برای یک مدل) و متد run فایل DatabaseSeeder.php (که از قبل موجود است) را در شاخه database/seeds بروزرسانی میکنیم.
برای ساخت فایلهای factory، این دستورات را در ترمینال اجرا میکنیم:
php artisan make:factory ContinentFactory --model=Continent
php artisan make:factory CountryFactory --model=Country
php artisan make:factory CityFactory --model=City
دستور بالا این فایلها را به ترتیب خواهد ساخت:
database/factories/ContinentFactory.php:
use Faker\Generator as Faker;
$factory->define(App\Continent::class, function (Faker $faker) {
return [
//
];
});
database/factories/CountryFactory.php:
use Faker\Generator as Faker;
$factory->define(App\Country::class, function (Faker $faker) {
return [
//
];
});
database/factories/CityFactory.php:
use Faker\Generator as Faker;
$factory->define(App\City::class, function (Faker $faker) {
return [
//
];
});
در آخر، بیایید فایل DatabaseSeeder.php را که تمام نمونههای تازه یک برنامه Laravel را دارد را بروزرسانی کنیم. ما از این فایل استفاده خواهیم کرد، تا سه ردیف به دیتابیس وارد کنیم. ما Africa، (از مدل قاره) South Africa (از مدل کشور) و Johannesburg (از مدل شهر) را وارد کرده، و همچنین رابطه آنها را مشخص میکنیم:
use Illuminate\Database\Seeder;
class DatabaseSeeder extends Seeder
{
/**
* نوشتن در دیتابیس را اجرا کن.
*
* @return void
*/
public function run()
{
factory(App\City::class)->create([
'name' => 'Johannesburg',
'country_id' => function(){
return factory(App\Country::class)->create([
'name' => 'South Africa',
'continent_id' => function(){
return factory(App\Continent::class)->create([
'name' => 'Africa' ])->id;
}
])->id;
}
]);
}
}
ما میتوانیم با اجرای این دستور در ترمینال، دیتابیس را پر کنیم:
php artisan db:seed --class=DatabaseSeeder
راهاندازی فایل Breadcrumb
ما باید فایلی به نام breadcrumbs.php در شاخه routes بسازی. این فایل، هر زمان که یک breadcrumb رندر میشود، ارجاع خواهد شد؛ زیرا به Laravel میگوید که چگونه اطلاعات breadcrumb را پردازش کند. بدون این فایل، Laravel هر زمان که با یک فراخوانی به یک تابع breadcrumb در viewها مواجه شود، یک خطا را برای ما به نمایش خواهد گذاشت.
اولین تابع breadcrumb که در فایل جدید به نام breadcrumbs.php که ساختیم تعریف میکنیم، تابع مربوط به صفحه خانه ما خواهد بود. Breadcrumb به نام «home» هر زمان که routeای با نام «home» بازدید شود، بارگذاری خواهد شد.
routes/breadcrumbs.php:
Breadcrumbs::register('home', function ($breadcrumbs) {
$breadcrumbs->push('Home', route('home'));
});
رندر کردن یک Breadcrumb استاتیک
در کد بالا، میبینیم که کلاس Breadcrumb برای فراخوانی یک متد استاتیک استفاده میشود. متد ثبت، یک Breadcrumb جدید با نام «home» است و یک closure را فراخوانی میکند. سپس این closure یک پارمتر $ breadcrumb را میگیرد و یک نمونه Breadcrumb جدید به همراه URL آن میفرستد.
$breadcrumbs->push('Home', route('home'));
«Home» در متد push، hard-code شده است و هر زمان که Breadcrumb مورد نظر بر روی view خانه، یا هر view دیگری که به Breadcrumb خانه برای نمایش یک زنجیره جهتیابی کامل نیاز دارد رندر شود، ظاهر خواهد شد.
route(‘home’)، یوآراِل «home» را بر میگراند و لینکی خواهد بود که هر زمان Home بر روی یک view جدید رندر میشود، به آن ختم خواهد شد.
در آخر، در کد view خانه، Breadcrumb را به این صورت رندر میکنیم:
resources/views/home.blade.php:
{{ Breadcrumbs::render('home') }}
متد render نام Breadcrumbای که باید بر روی view خانه نمایش داده شود را دریافت میکند. در زنجیرههای جهتیابی پیچیدهتر که نوعی رابطه دیتابیس را شامل میشوند، تابع render هر تعداد نمونه Models که برای رندر کردن زنجیره جهتیابی Breadcrumb مورد نیاز هستند را به عنوان آرگومان قبول خواهد کرد.
در ادامه بیشتر راجب این مسئله صحبت خواهیم کرد.
رندر کردن یک Breadcrumb دینامیک
وقتی که یک وباپلیکیشن داریم، و صفات دقیق صفحاتی که کاربرها بازدید خواهند کرد را نمیدانیم، چه اتفاقی میافتد؟ برای مثال، وبسایتی داریم که اطلاعاتی درباره قارهها، کشورها و شهرها را نمایش میدهد. ما همیشه نمیتوانیم به فایل routes/breadcrumbs.php رفته و نام هر قاره، کشور یا شهری را hard-codeکنیم؛ زیرا هیچ وقت نمیدانیم که کاربر کدام مورد را بازدید خواهد کرد.
در جهت ممکن کردن رندر کردن breadcrumb به صورت دینامیک و realtime، میتوانیم فایل routes/breadcrumbs.php را به گونهای بنویسیم که به صورت موثر با مدلهای دینامیک در برنامه ما کار کند و همینطور که کاربرها لایههای عمیقتری از وباپلیکیشن ما را مرور میکنند، زنجیره جهتیابی breadcrumb را بسازیم.
همانطور که پیشتر گفتیم، مدل قاره چندین کشور را دارد (continent model hasMany countries) و مدل کشور چندین شهر را دارد. (country model hasMany cities) با دانستن این مسئله، باید بتوانیم یک نمایش breadcrumb به دست بیاوریم که درست به مانند این باشد که ما Johannesburg از آفریقا را بازدید میکنیم، که در آن South Africa به Africa تعلق دارد.
برای ساخت زنجیره جهتیابی Breadcrumbها با استفاده از رابطههای Model به مانند آنچه در بالا میبینیم، با نوشتن کد مربوط به ثبت breadcrumb با نام continent شروع میکنیم:
routes/breadcrumbs.php:
Breadcrumbs::register('continent', function ($breadcrumbs, $continent) {
$breadcrumbs->parent('home');
$breadcrumbs->push($continent->name, route('continent', ['name' => $continent->name]));
});
ما یک breadcrumb به نام continent ثبت کرده، و یک closure را منتقل میکنیم. این closure یک پارامتر جدید به نام $continent (که توسط view مورد نظر به صورت realtime عرضه میشود) را دریافت میکند. سپس، breadcrumb به نام home به والد خود اختصاص داده میشود و در آخر، نام و یوآراِل breadcrumb با نام $continent وارد میشود.
برای رندر کردن breadcrumb در کد view قاره، این قطعه کد را مینویسیم:
resources/views/continent.blade.php:
{{ Breadcrumbs::render('continent', $continent) }}
متد render نام breadcrumb را به عنوان آرگومان اول دریافت میکند. این متد همچنین آرگومان $continent را دریافت میکند که در resolve کردن نمونه مدل قاره به ویژگی پایهاش مانند نام و URL کمک خواهد کرد.
بر روی دستگاه خود و در آدرس http://127.0.0.1:8000/continent/africa، صفحه قاره Africa را به همراه نمایش breadcrumb زیر، دریافت خواهید کرد:
رندر کردن یک زنجیره جهتیابی کامل
ما به کد مربوط به رندر کردن breadcrumbهای تکی نگاه داشتهایم. حال بیایید به کد مربوط به رندر کردن یک زنجیره breadcrumb کامل نگاهی داشته باشیم. ما یک صفحه شهر که به یک صفحه کشور مرتبط است، یک صفحه قاره و در نهایت یک صفحه خانه را رندر خواهیم کرد.
این کد نهایی برای فایل routes/breadcrumbs.php است. (البته برای این مقاله. این کد میتواند بر حسب هدفی که دارید، بزرگتر شود.)
routes/breadcrumbs.php:
Breadcrumbs::register('home', function ($breadcrumbs) {
$breadcrumbs->push('Home', route('home'));
});
Breadcrumbs::register('continent', function ($breadcrumbs, $continent) {
$breadcrumbs->parent('home');
$breadcrumbs->push($continent->name, route('continent', ['name' => $continent->name]));
});
Breadcrumbs::register('country', function ($breadcrumbs, $continent, $country) {
$breadcrumbs->parent('continent', $continent);
$breadcrumbs->push($country->name, route('country', ['name' => $country->name]));
});
Breadcrumbs::register('city', function ($breadcrumbs, $continent, $country, $city) {
$breadcrumbs->parent('country', $continent, $country);
$breadcrumbs->push($city->name, route('city', ['name' => $city->name]));
});
ما دو breadcrumb دیگر را اضافه کردهایم: country و city. ما به این breadcrumbها نیاز خواهیم داشت، تا تمام breadcrumbها بتوانند هر زمان که یک view برای یک زنجیره جهتیابی فراخوانی میکند که شامل بیش از یک breadcrumb است، به صورت موثر با یکدیگر ارتباط داشته باشند.
متدهای مربوط به ثبت و وارد کردن breadcrumbهای country و city، مشابه متدهای مربوط به continent هستند؛ به همین دلیل به جزئیات آنها وارد نخواهیم شد.
برای رندر کردن breadcrumb در کد view برای صفحه city، به سادگی این قطعه کد را از موقعیت مورد نظر وارد میکنیم:
resources/views/city.blade.php:
{{ Breadcrumbs::render('city', $city->country->continent, $city->country, $city) }}
متد render در اینجا نام breadcrumb را به عنوان آرگومان اول دریافت میکند. سپس، یک آرگومان به نام $city->country->continent را دریافت میکند که به یک نمونه مدل continent ارزیابی خواهد شد. این متد همچنین یک آرگومان به نام $city->country دریافت میکند که کشوری است که شهر به آن تعلق دارد (belongsTo) و در نهایت، یک نمونه $city.
در اینجا کار ما به پایان میرسد. ما جهتیابی breadcrumb عملکردی و دینامیک خود را در چند خط کد به طور کامل ساختهایم.
نتیجه گیری
در دنیایی که کاربران سرویسهای اینترنت به دنبال راحتترین محصولات هستند، اضافه کردن breadcrumb به وبسایتها، یک کار کاملا صحیح از طرف توسعه دهندگان است. این مقاله این مسئله را به ما آموخته است. با تنها چند خط کد، میتوانیم یک سیستم جهتیابی کاملا عملکردی بسازیم. کارهای بسیار بیشتری هم هستند که میتوانند با استفاده از پکیج Laravel Breadcrumb انجام شوند، (برای مثال میتوانیم استایل و طرح را طبق میل خود تحت تاثیر قرار دهیم) پس بروید و حداکثر قدرت آن را آزمایش کنید.
دیدگاه و پرسش
در حال دریافت نظرات از سرور، لطفا منتظر بمانید
در حال دریافت نظرات از سرور، لطفا منتظر بمانید