You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Web-Dev-For-Beginners/translations/fa/7-bank-project/3-data
Lee Stott 2daab5271b
Update Quiz Link
3 weeks ago
..
README.md Update Quiz Link 3 weeks ago
assignment.md 🌐 Update translations via Co-op Translator 4 weeks ago

README.md

ساخت اپلیکیشن بانکی قسمت ۳: روش‌های دریافت و استفاده از داده‌ها

آزمون پیش از درس

آزمون پیش از درس

مقدمه

در قلب هر اپلیکیشن وب، داده‌ها قرار دارند. داده‌ها می‌توانند اشکال مختلفی داشته باشند، اما هدف اصلی آن‌ها همیشه نمایش اطلاعات به کاربر است. با پیچیده‌تر و تعاملی‌تر شدن اپلیکیشن‌های وب، نحوه دسترسی و تعامل کاربر با اطلاعات به یکی از بخش‌های کلیدی توسعه وب تبدیل شده است.

در این درس، یاد می‌گیریم که چگونه داده‌ها را به صورت غیرهمزمان از سرور دریافت کنیم و از این داده‌ها برای نمایش اطلاعات در یک صفحه وب بدون بارگذاری مجدد HTML استفاده کنیم.

پیش‌نیاز

برای این درس، باید بخش فرم ورود و ثبت‌نام اپلیکیشن وب را ساخته باشید. همچنین باید Node.js را نصب کرده و API سرور را به صورت محلی اجرا کنید تا بتوانید داده‌های حساب کاربری را دریافت کنید.

برای اطمینان از اینکه سرور به درستی اجرا می‌شود، می‌توانید این دستور را در ترمینال اجرا کنید:

curl http://localhost:5000/api
# -> should return "Bank API v1.0.0" as a result

AJAX و دریافت داده‌ها

وب‌سایت‌های سنتی زمانی که کاربر یک لینک را انتخاب می‌کند یا داده‌ای را از طریق فرم ارسال می‌کند، محتوای نمایش داده شده را با بارگذاری مجدد کل صفحه HTML به‌روزرسانی می‌کنند. هر بار که نیاز به بارگذاری داده‌های جدید باشد، سرور وب یک صفحه HTML کاملاً جدید بازمی‌گرداند که باید توسط مرورگر پردازش شود، این فرآیند اقدام فعلی کاربر را قطع کرده و تعاملات را در طول بارگذاری محدود می‌کند. این جریان کاری همچنین به عنوان اپلیکیشن چندصفحه‌ای یا MPA شناخته می‌شود.

جریان به‌روزرسانی در یک اپلیکیشن چندصفحه‌ای

با پیچیده‌تر و تعاملی‌تر شدن اپلیکیشن‌های وب، تکنیک جدیدی به نام AJAX (جاوااسکریپت و XML غیرهمزمان) معرفی شد. این تکنیک به اپلیکیشن‌های وب اجازه می‌دهد که داده‌ها را به صورت غیرهمزمان از سرور ارسال و دریافت کنند، بدون نیاز به بارگذاری مجدد صفحه HTML، که منجر به به‌روزرسانی‌های سریع‌تر و تعاملات روان‌تر کاربر می‌شود. زمانی که داده‌های جدید از سرور دریافت می‌شود، صفحه HTML فعلی نیز می‌تواند با استفاده از API DOM توسط جاوااسکریپت به‌روزرسانی شود. با گذشت زمان، این رویکرد به چیزی تبدیل شده که اکنون به عنوان اپلیکیشن تک‌صفحه‌ای یا SPA شناخته می‌شود.

جریان به‌روزرسانی در یک اپلیکیشن تک‌صفحه‌ای

زمانی که AJAX برای اولین بار معرفی شد، تنها API موجود برای دریافت داده‌ها به صورت غیرهمزمان XMLHttpRequest بود. اما مرورگرهای مدرن اکنون API قدرتمندتر و راحت‌تر Fetch را پیاده‌سازی کرده‌اند که از وعده‌ها استفاده می‌کند و برای دستکاری داده‌های JSON مناسب‌تر است.

در حالی که همه مرورگرهای مدرن از Fetch API پشتیبانی می‌کنند، اگر می‌خواهید اپلیکیشن وب شما روی مرورگرهای قدیمی یا قدیمی‌تر کار کند، همیشه بهتر است ابتدا جدول سازگاری در caniuse.com را بررسی کنید.

وظیفه

در درس قبلی فرم ثبت‌نام را برای ایجاد حساب کاربری پیاده‌سازی کردیم. اکنون کدی اضافه می‌کنیم تا با استفاده از یک حساب موجود وارد شوید و داده‌های آن را دریافت کنید. فایل app.js را باز کرده و یک تابع جدید به نام login اضافه کنید:

async function login() {
  const loginForm = document.getElementById('loginForm')
  const user = loginForm.user.value;
}

در اینجا ابتدا عنصر فرم را با استفاده از getElementById() دریافت می‌کنیم و سپس نام کاربری را از ورودی با loginForm.user.value می‌گیریم. هر کنترل فرم را می‌توان با نام آن (که در HTML با استفاده از ویژگی name تنظیم شده است) به عنوان یک ویژگی از فرم دسترسی داشت.

به همان شیوه‌ای که برای ثبت‌نام انجام دادیم، یک تابع دیگر برای انجام درخواست سرور ایجاد می‌کنیم، اما این بار برای دریافت داده‌های حساب کاربری:

async function getAccount(user) {
  try {
    const response = await fetch('//localhost:5000/api/accounts/' + encodeURIComponent(user));
    return await response.json();
  } catch (error) {
    return { error: error.message || 'Unknown error' };
  }
}

ما از API fetch برای درخواست داده‌ها به صورت غیرهمزمان از سرور استفاده می‌کنیم، اما این بار به هیچ پارامتر اضافی به جز URL نیاز نداریم، زیرا فقط در حال پرس‌وجوی داده‌ها هستیم. به طور پیش‌فرض، fetch یک درخواست HTTP GET ایجاد می‌کند، که همان چیزی است که در اینجا به دنبال آن هستیم.

encodeURIComponent() یک تابع است که کاراکترهای خاص را برای URL فرار می‌دهد. اگر این تابع را فراخوانی نکنیم و مستقیماً مقدار user را در URL استفاده کنیم، چه مشکلاتی ممکن است پیش بیاید؟

اکنون تابع login خود را به‌روزرسانی می‌کنیم تا از getAccount استفاده کند:

async function login() {
  const loginForm = document.getElementById('loginForm')
  const user = loginForm.user.value;
  const data = await getAccount(user);

  if (data.error) {
    return console.log('loginError', data.error);
  }

  account = data;
  navigate('/dashboard');
}

ابتدا، از آنجا که getAccount یک تابع غیرهمزمان است، باید آن را با کلمه کلیدی await مطابقت دهیم تا منتظر نتیجه سرور بمانیم. همانند هر درخواست سرور، باید با موارد خطا نیز برخورد کنیم. فعلاً فقط یک پیام لاگ برای نمایش خطا اضافه می‌کنیم و بعداً به آن بازمی‌گردیم.

سپس باید داده‌ها را جایی ذخیره کنیم تا بتوانیم بعداً از آن برای نمایش اطلاعات داشبورد استفاده کنیم. از آنجا که متغیر account هنوز وجود ندارد، یک متغیر سراسری برای آن در بالای فایل خود ایجاد می‌کنیم:

let account = null;

پس از ذخیره داده‌های کاربر در یک متغیر، می‌توانیم از صفحه ورود به صفحه داشبورد با استفاده از تابع navigate() که قبلاً داریم، حرکت کنیم.

در نهایت، باید تابع login خود را زمانی که فرم ورود ارسال می‌شود، فراخوانی کنیم، با اصلاح HTML:

<form id="loginForm" action="javascript:login()">

تست کنید که همه چیز به درستی کار می‌کند، با ثبت یک حساب جدید و تلاش برای ورود با همان حساب.

قبل از ادامه به بخش بعدی، می‌توانیم تابع register را نیز کامل کنیم، با اضافه کردن این کد در انتهای تابع:

account = result;
navigate('/dashboard');

آیا می‌دانستید که به طور پیش‌فرض، فقط می‌توانید APIهای سرور را از همان دامنه و پورت که صفحه وب را مشاهده می‌کنید، فراخوانی کنید؟ این یک مکانیزم امنیتی است که توسط مرورگرها اجرا می‌شود. اما صبر کنید، اپلیکیشن وب ما روی localhost:3000 اجرا می‌شود در حالی که API سرور روی localhost:5000 اجرا می‌شود، چرا کار می‌کند؟ با استفاده از تکنیکی به نام اشتراک منابع بین‌مبدأ (CORS)، امکان انجام درخواست‌های HTTP بین‌مبدأ وجود دارد اگر سرور هدرهای خاصی به پاسخ اضافه کند و استثناهایی برای دامنه‌های خاص اجازه دهد.

با گرفتن این درس درباره APIها بیشتر بیاموزید.

به‌روزرسانی HTML برای نمایش داده‌ها

اکنون که داده‌های کاربر را داریم، باید HTML موجود را برای نمایش آن به‌روزرسانی کنیم. ما قبلاً می‌دانیم که چگونه یک عنصر را از DOM با استفاده از مثلاً document.getElementById() دریافت کنیم. پس از داشتن یک عنصر پایه، اینجا چند API وجود دارد که می‌توانید از آن‌ها برای تغییر یا اضافه کردن عناصر فرزند به آن استفاده کنید:

  • با استفاده از ویژگی textContent می‌توانید متن یک عنصر را تغییر دهید. توجه داشته باشید که تغییر این مقدار تمام فرزندان عنصر (اگر وجود داشته باشد) را حذف کرده و آن را با متن ارائه شده جایگزین می‌کند. به همین دلیل، این روش همچنین یک روش کارآمد برای حذف تمام فرزندان یک عنصر خاص با اختصاص یک رشته خالی '' به آن است.

  • با استفاده از document.createElement() همراه با روش append() می‌توانید یک یا چند عنصر فرزند جدید ایجاد و پیوست کنید.

با استفاده از ویژگی innerHTML یک عنصر، همچنین می‌توان محتوای HTML آن را تغییر داد، اما این ویژگی باید اجتناب شود زیرا در برابر حملات اسکریپت‌نویسی بین‌سایتی (XSS) آسیب‌پذیر است.

وظیفه

قبل از ادامه به صفحه داشبورد، یک کار دیگر باید در صفحه ورود انجام دهیم. در حال حاضر، اگر سعی کنید با یک نام کاربری که وجود ندارد وارد شوید، یک پیام در کنسول نمایش داده می‌شود اما برای یک کاربر عادی هیچ چیزی تغییر نمی‌کند و نمی‌دانید چه اتفاقی افتاده است.

بیایید یک عنصر جایگزین در فرم ورود اضافه کنیم که در صورت نیاز بتوانیم یک پیام خطا نمایش دهیم. یک مکان خوب درست قبل از دکمه ورود <button> خواهد بود:

...
<div id="loginError"></div>
<button>Login</button>
...

این عنصر <div> خالی است، به این معنی که هیچ چیزی روی صفحه نمایش داده نمی‌شود تا زمانی که محتوایی به آن اضافه کنیم. همچنین به آن یک id می‌دهیم تا بتوانیم به راحتی با جاوااسکریپت آن را دریافت کنیم.

به فایل app.js بازگردید و یک تابع کمکی جدید به نام updateElement ایجاد کنید:

function updateElement(id, text) {
  const element = document.getElementById(id);
  element.textContent = text;
}

این یکی کاملاً ساده است: با توجه به یک id عنصر و متن، متن محتوای عنصر DOM با id مطابقت را به‌روزرسانی می‌کند. بیایید از این روش به جای پیام خطای قبلی در تابع login استفاده کنیم:

if (data.error) {
  return updateElement('loginError', data.error);
}

اکنون اگر سعی کنید با یک حساب نامعتبر وارد شوید، باید چیزی شبیه به این را ببینید:

تصویری که پیام خطا را در هنگام ورود نمایش می‌دهد

اکنون متن خطا به صورت بصری نمایش داده می‌شود، اما اگر آن را با یک صفحه‌خوان امتحان کنید، متوجه می‌شوید که هیچ چیزی اعلام نمی‌شود. برای اینکه متنی که به صورت پویا به صفحه اضافه می‌شود توسط صفحه‌خوان‌ها اعلام شود، باید از چیزی به نام منطقه زنده استفاده کند. در اینجا از نوع خاصی از منطقه زنده به نام هشدار استفاده می‌کنیم:

<div id="loginError" role="alert"></div>

همین رفتار را برای خطاهای تابع register پیاده‌سازی کنید (فراموش نکنید که HTML را به‌روزرسانی کنید).

نمایش اطلاعات در داشبورد

با استفاده از همان تکنیک‌هایی که به تازگی دیده‌ایم، همچنین به نمایش اطلاعات حساب در صفحه داشبورد می‌پردازیم.

این چیزی است که یک شیء حساب دریافت شده از سرور به نظر می‌رسد:

{
  "user": "test",
  "currency": "$",
  "description": "Test account",
  "balance": 75,
  "transactions": [
    { "id": "1", "date": "2020-10-01", "object": "Pocket money", "amount": 50 },
    { "id": "2", "date": "2020-10-03", "object": "Book", "amount": -10 },
    { "id": "3", "date": "2020-10-04", "object": "Sandwich", "amount": -5 }
  ],
}

توجه: برای راحتی بیشتر، می‌توانید از حساب test از پیش موجود که قبلاً با داده‌ها پر شده است استفاده کنید.

وظیفه

بیایید با جایگزینی بخش "موجودی" در HTML برای اضافه کردن عناصر جایگزین شروع کنیم:

<section>
  Balance: <span id="balance"></span><span id="currency"></span>
</section>

همچنین یک بخش جدید درست زیر آن برای نمایش توضیحات حساب اضافه می‌کنیم:

<h2 id="description"></h2>

از آنجا که توضیحات حساب به عنوان عنوانی برای محتوای زیر آن عمل می‌کند، به صورت معنایی به عنوان یک عنوان علامت‌گذاری شده است. درباره اینکه چگونه ساختار عنوان برای دسترسی مهم است بیشتر بیاموزید و به صورت انتقادی به صفحه نگاه کنید تا تعیین کنید چه چیز دیگری می‌تواند یک عنوان باشد.

سپس، یک تابع جدید در app.js برای پر کردن جایگزین ایجاد می‌کنیم:

function updateDashboard() {
  if (!account) {
    return navigate('/login');
  }

  updateElement('description', account.description);
  updateElement('balance', account.balance.toFixed(2));
  updateElement('currency', account.currency);
}

ابتدا بررسی می‌کنیم که داده‌های حساب مورد نیاز را داریم قبل از اینکه جلوتر برویم. سپس از تابع updateElement() که قبلاً ایجاد کردیم برای به‌روزرسانی HTML استفاده می‌کنیم.

برای زیباتر کردن نمایش موجودی، از روش toFixed(2) استفاده می‌کنیم تا نمایش مقدار با ۲ رقم بعد از نقطه اعشار را مجبور کنیم.

اکنون باید تابع updateDashboard() خود را هر بار که صفحه داشبورد بارگذاری می‌شود، فراخوانی کنیم. اگر قبلاً تکلیف درس ۱ را تمام کرده‌اید، این باید ساده باشد، در غیر این صورت می‌توانید از پیاده‌سازی زیر استفاده کنید.

این کد را به انتهای تابع updateRoute() اضافه کنید:

if (typeof route.init === 'function') {
  route.init();
}

و تعریف مسیرها را با این کد به‌روزرسانی کنید:

const routes = {
  '/login': { templateId: 'login' },
  '/dashboard': { templateId: 'dashboard', init: updateDashboard }
};

با این تغییر، هر بار که صفحه داشبورد نمایش داده می‌شود، تابع updateDashboard() فراخوانی می‌شود. پس از ورود، باید بتوانید موجودی حساب، ارز و توضیحات را مشاهده کنید.

ایجاد ردیف‌های جدول به صورت پویا با قالب‌های HTML

در درس اول از قالب‌های HTML همراه با روش appendChild() برای پیاده‌سازی ناوبری در اپلیکیشن خود استفاده کردیم. قالب‌ها همچنین می‌توانند کوچک‌تر باشند و برای پر کردن بخش‌های تکراری یک صفحه به صورت پویا استفاده شوند.

از رویکرد مشابهی برای نمایش لیست تراکنش‌ها در جدول HTML استفاده خواهیم کرد.

وظیفه

یک قالب جدید در <body> HTML اضافه کنید:

<template id="transaction">
  <tr>
    <td></td>
    <td></td>
    <td></td>
  </tr>
</template>

این قالب یک ردیف جدول واحد را نشان می‌دهد، با ۳ ستونی که می‌خواهیم پر کنیم: تاریخ، موضوع و مقدار یک تراکنش.

سپس، این ویژگی id را به عنصر <tbody> جدول در قالب داشبورد اضافه کنید تا پیدا کردن آن با جاوااسکریپت آسان‌تر شود:

<tbody id="transactions"></tbody>

HTML ما آماده است، بیایید به کد جاوااسکریپت برویم و یک تابع جدید به نام createTransactionRow ایجاد کنیم:

function createTransactionRow(transaction) {
  const template = document.getElementById('transaction');
  const transactionRow = template.content.cloneNode(true);
  const tr = transactionRow.querySelector('tr');
  tr.children[0].textContent = transaction.date;
  tr.children[1].textContent = transaction.object;
  tr.children[2].textContent = transaction.amount.toFixed(2);
  return transactionRow;
}

این تابع دقیقاً همان کاری را انجام می‌دهد که نامش نشان می‌دهد: با استفاده از قالبی که قبلاً ایجاد کردیم، یک ردیف جدول جدید ایجاد می‌کند و محتوای آن را با استفاده از داده‌های تراکنش پر می‌کند. از این تابع در updateDashboard() برای پر کردن جدول استفاده خواهیم کرد:

const transactionsRows = document.createDocumentFragment();
for (const transaction of account.transactions) {
  const transactionRow = createTransactionRow(transaction);
  transactionsRows.appendChild(transactionRow);
}
updateElement('transactions', transactionsRows);

در اینجا از روش document.createDocumentFragment() استفاده می‌کنیم که یک قطعه جدید DOM ایجاد می‌کند که می‌توانیم روی آن کار کنیم، قبل از اینکه در نهایت آن را به جدول HTML خود پیوست کنیم.

هنوز یک کار دیگر باید انجام دهیم تا این کد کار کند، زیرا تابع updateElement() ما در حال حاضر فقط از محتوای متنی پشتیبانی می‌کند. بیایید کد آن را کمی تغییر دهیم:

function updateElement(id, textOrNode) {
  const element = document.getElementById(id);
  element.textContent = ''; // Removes all children
  element.append(textOrNode);
}

از روش append() استفاده می‌کنیم زیرا به ما اجازه می‌دهد که متن یا گره‌های DOM را به یک عنصر والد پیوست کنیم، که برای همه موارد استفاده ما مناسب است. اگر از حساب کاربری test برای ورود استفاده کنید، اکنون باید لیست تراکنش‌ها را در داشبورد مشاهده کنید 🎉.


🚀 چالش

با همکاری یکدیگر، صفحه داشبورد را طوری طراحی کنید که شبیه یک اپلیکیشن واقعی بانکی به نظر برسد. اگر قبلاً اپلیکیشن خود را طراحی کرده‌اید، سعی کنید از media queries برای ایجاد یک طراحی واکنش‌گرا استفاده کنید که به خوبی روی دستگاه‌های دسکتاپ و موبایل کار کند.

در اینجا یک نمونه از صفحه داشبورد طراحی‌شده آورده شده است:

تصویری از نتیجه نمونه داشبورد پس از طراحی

آزمون پس از درس

آزمون پس از درس

تکلیف

بازنویسی و توضیح کد خود

سلب مسئولیت:
این سند با استفاده از سرویس ترجمه هوش مصنوعی Co-op Translator ترجمه شده است. در حالی که ما تلاش می‌کنیم دقت را حفظ کنیم، لطفاً توجه داشته باشید که ترجمه‌های خودکار ممکن است حاوی خطاها یا نادرستی‌هایی باشند. سند اصلی به زبان اصلی آن باید به عنوان منبع معتبر در نظر گرفته شود. برای اطلاعات حساس، ترجمه حرفه‌ای انسانی توصیه می‌شود. ما هیچ مسئولیتی در قبال سوءتفاهم‌ها یا تفسیرهای نادرست ناشی از استفاده از این ترجمه نداریم.