WiX/mag

ویکس مگ / مجله آموزشی ویکس سِوِن

مدیریت رویدادها (Event Handling) در ری‌اکت: راهنمای جامع

پیمان باقری

پیمان باقری

برنامه نویس و متخصص وب

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

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

رویداد در ری‌اکت چیست؟

در ری‌اکت، رویدادها (Events) روش‌هایی برای برقراری تعاملات کاربر با اپلیکیشن هستند، مشابه رویدادهای DOM در جاوااسکریپت معمولی. با این حال، ری‌اکت برای مدیریت این تعاملات، سینتکس و مکانیزم خاص خود را ارائه می‌دهد که مزایای متعددی را به همراه دارد.

نحوه تعریف و استفاده از رویدادها

تعریف سینتکس خاص: برخلاف جاوااسکریپت معمولی که از نام‌هایی مانند onclick و onsubmit استفاده می‌کند، ری‌اکت از نام‌های کامل‌تر و CamelCase مانند onClick و onSubmit بهره می‌گیرد. این تفاوت باعث خوانایی بیشتر کد و هماهنگی با سایر استانداردهای JSX می‌شود.

مثال:

<button onClick={handleClick}>کلیک کن</button>

در اینجا، handleClick یک تابع جاوااسکریپتی است که هنگام کلیک روی دکمه اجرا می‌شود.

ویژگی‌های رویدادها در ری‌اکت

  1. تعریف به صورت پراپرتی JSX: رویدادها مستقیماً به عنوان پراپرتی‌هایی در تگ‌های JSX تعریف می‌شوند. این رویکرد اجازه می‌دهد که رویدادها به صورت توابع کامپوننت به سادگی متصل شوند.
  2. سازگاری کامل با SyntheticEvent: در ری‌اکت، رویدادها با یک لایه انتزاعی به نام SyntheticEvent پیچیده شده‌اند. این لایه:
  3. تجربه‌ای مشابه با رویدادهای بومی مرورگر ارائه می‌دهد.
  4. سازگاری با مرورگرهای مختلف را تضمین می‌کند.
  5. کارایی بیشتری برای مدیریت حافظه فراهم می‌آورد.

چرایی استفاده از رویدادها در ری‌اکت

  • ایجاد تعامل پویا در اپلیکیشن.
  • تسهیل مدیریت رخدادهای پیچیده.
  • بهینه‌سازی و سازگاری کد برای اجرا در مرورگرهای مختلف.

این سینتکس خاص و استاندارد باعث می‌شود که کدهای ری‌اکت خواناتر، ساختاریافته‌تر و مقیاس‌پذیرتر باشند.

 

مدیریت رویدادها در کامپوننت‌های تابعی

در کامپوننت‌های تابعی ری‌اکت، مدیریت رویدادها با استفاده از سینتکس ساده و مدرن جاوااسکریپت انجام می‌شود. این روش، به‌ویژه با معرفی هوک‌ها مانند useState و useEffect، انعطاف بیشتری برای مدیریت تعاملات فراهم کرده است.

نحوه تعریف رویدادها در کامپوننت‌های تابعی

  1. تعریف تابع برای مدیریت رویداد: در کامپوننت‌های تابعی، رویدادها معمولاً به یک تابع ساده اختصاص داده می‌شوند. این تابع معمولاً در همان کامپوننت تعریف می‌شود و به عنوان پراپرتی به المان JSX متصل می‌گردد.
import React, { useState } from 'react';

function App() {
  const [count, setCount] = useState(0);

  const handleClick = () => {
    setCount(count + 1);
  };

  return (
    <button onClick={handleClick}>
      کلیک‌ها: {count}
    </button>
  );
}

در این مثال، تابع handleClick هنگام کلیک روی دکمه فراخوانی می‌شود و وضعیت (state) شمارنده را به‌روزرسانی می‌کند.

  1. استفاده از توابع درجا (Inline Functions): به‌جای تعریف جداگانه تابع، می‌توان توابع را به صورت مستقیم در پراپرتی رویداد تعریف کرد.
<button onClick={() => alert('کلیک شد!')}>کلیک کن</button>

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

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

  1. به‌روزرسانی وضعیت با useState: هوک useState به شما اجازه می‌دهد که وضعیت کامپوننت را به‌روزرسانی کنید. برای مثال:
const [text, setText] = useState('');

const handleChange = (event) => {
  setText(event.target.value);
};

return <input type="text" value={text} onChange={handleChange} />;

در اینجا، هر بار که کاربر متنی را وارد می‌کند، رویداد onChange مقدار text را به‌روزرسانی می‌کند.

  1. تعامل با رویدادهای خارج از کامپوننت با useEffect: اگر نیاز باشد که به رویدادهایی خارج از المان JSX گوش دهید (مانند اسکرول یا رویدادهای صفحه کلید)، هوک useEffect می‌تواند کمک کند:
import React, { useState, useEffect } from 'react';

function App() {
  const [key, setKey] = useState('');

  useEffect(() => {
    const handleKeyPress = (event) => {
      setKey(event.key);
    };
    window.addEventListener('keypress', handleKeyPress);

    return () => {
      window.removeEventListener('keypress', handleKeyPress);
    };
  }, []);

  return <p>آخرین کلید فشرده شده: {key}</p>;
}

این مثال نشان می‌دهد که چگونه با استفاده از useEffect می‌توانید رویدادهایی را که به المان‌های خاصی متصل نیستند، مدیریت کنید.

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

  • سینتکس ساده‌تر و قابل‌فهم‌تر: کامپوننت‌های تابعی نیازی به مدیریت پیچیده مانند بایند کردن (bind) ندارند.
  • استفاده از توابع خالص: تابع‌ها قابل پیش‌بینی و قابل تست هستند.
  • یکپارچگی با هوک‌ها: هوک‌ها امکانات بیشتری برای مدیریت وضعیت و تعاملات پیچیده فراهم می‌کنند.

با این روش‌ها، کامپوننت‌های تابعی انعطاف‌پذیری و سادگی بیشتری را برای مدیریت رویدادها ارائه می‌دهند و گزینه مناسبی برای توسعه‌دهندگان مدرن به شمار می‌روند.

 

مدیریت رویدادها در کامپوننت‌های کلاسی

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

تعریف و اتصال متدهای مدیریت رویداد در کلاس‌ها

در کامپوننت‌های کلاسی، متدهای مدیریت رویداد معمولاً به صورت توابعی در داخل کلاس تعریف شده و به المان‌های JSX متصل می‌شوند. به عنوان مثال:

import React, { Component } from 'react';

class App extends Component {
  handleClick() {
    alert('دکمه کلیک شد!');
  }

  render() {
    return (
      <button onClick={this.handleClick}>
        کلیک کن
      </button>
    );
  }
}

در این مثال، متد handleClick به رویداد onClick دکمه متصل شده است. اما این روش ممکن است مشکلاتی ایجاد کند، مانند دسترسی نادرست به مقدار this در متدها.

مشکل دسترسی به this و حل آن

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

الف) استفاده از bind در سازنده

constructor(props) {
  super(props);
  this.handleClick = this.handleClick.bind(this);
}

با این روش، متد handleClick به نمونه کلاس متصل می‌شود و دسترسی به this صحیح خواهد بود.

ب) استفاده از توابع فلش

توابع فلش به صورت خودکار this را به شی فعلی بایند می‌کنند. بنابراین می‌توان به شکل زیر متدها را تعریف کرد:

handleClick = () => {
  alert('دکمه کلیک شد!');
};

مثال پیشرفته‌تر: مدیریت وضعیت در کلاس‌ها با رویدادها

یک مثال کاربردی که تغییر وضعیت را با یک رویداد مدیریت می‌کند:

import React, { Component } from 'react';

class Counter extends Component {
  constructor(props) {
    super(props);
    this.state = { count: 0 };
  }

  increment = () => {
    this.setState((prevState) => ({
      count: prevState.count + 1
    }));
  };

  render() {
    return (
      <div>
        <p>شمارنده: {this.state.count}</p>
        <button onClick={this.increment}>افزایش</button>
      </div>
    );
  }
}

در این مثال:

  • متد increment وضعیت (state) را تغییر می‌دهد.
  • setState به صورت غیرهم‌زمان کار می‌کند، بنابراین از تابعی برای دسترسی به وضعیت فعلی استفاده شده است.

رویدادهای سفارشی

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

class Parent extends Component {
  handleChildClick = (message) => {
    alert(message);
  };

  render() {
    return <Child onButtonClick={this.handleChildClick} />;
  }
}

class Child extends Component {
  render() {
    return (
      <button onClick={() => this.props.onButtonClick('فرزند کلیک شد!')}>
        کلیک فرزند
      </button>
    );
  }
}

مزایای مدیریت رویدادها در کامپوننت‌های کلاسی

  • دسترسی آسان به state و متدهای کلاس.
  • امکان استفاده از متدهای lifecycle برای مدیریت پیچیده‌تر وضعیت و رویدادها.

محدودیت‌ها و جایگزین‌های مدرن

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

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

 

ارسال فرم‌ها و مدیریت رویدادهای ارسال (Submit)

در برنامه‌های وب، ارسال فرم یکی از رایج‌ترین تعاملات کاربر است. ری‌ اکت با ارائه راهکارهایی ساده و قدرتمند، مدیریت رویدادهای ارسال فرم را امکان‌پذیر کرده است. رویداد onSubmit در ری‌اکت برای مدیریت ارسال فرم‌ها استفاده می‌شود و با سینتکس JSX به سادگی پیاده‌سازی می‌شود.

اتصال رویداد onSubmit به فرم

برای مدیریت ارسال فرم در ری‌اکت، می‌توانید یک تابع هندلر تعریف کرده و آن را به رویداد onSubmit متصل کنید. به عنوان مثال:

function App() {
  const handleSubmit = (event) => {
    event.preventDefault(); // جلوگیری از رفتار پیش‌فرض مرورگر
    console.log('فرم ارسال شد!');
  };

  return (
    <form onSubmit={handleSubmit}>
      <input type="text" placeholder="نام خود را وارد کنید" />
      <button type="submit">ارسال</button>
    </form>
  );
}

توضیحات:

  1. جلوگیری از رفتار پیش‌فرض مرورگر: با استفاده از preventDefault()، از ریفرش شدن صفحه جلوگیری می‌شود.
  2. مدیریت داده‌های فرم: داده‌ها را می‌توان از عناصر فرم (مانند input) دریافت و پردازش کرد.

ذخیره‌سازی داده‌های فرم با State

برای مدیریت داده‌های فرم در زمان ارسال، معمولاً از state استفاده می‌شود. به عنوان نمونه:

import React, { useState } from 'react';

function App() {
  const [formData, setFormData] = useState({ name: '', email: '' });

  const handleChange = (event) => {
    const { name, value } = event.target;
    setFormData((prevData) => ({
      ...prevData,
      [name]: value,
    }));
  };

  const handleSubmit = (event) => {
    event.preventDefault();
    console.log('اطلاعات فرم:', formData);
  };

  return (
    <form onSubmit={handleSubmit}>
      <input
        type="text"
        name="name"
        placeholder="نام"
        value={formData.name}
        onChange={handleChange}
      />
      <input
        type="email"
        name="email"
        placeholder="ایمیل"
        value={formData.email}
        onChange={handleChange}
      />
      <button type="submit">ارسال</button>
    </form>
  );
}

توضیحات:

  1. کنترل ورودی‌ها با State: با استفاده از ویژگی‌های value و onChange، مقادیر ورودی‌ها در state ذخیره می‌شوند.
  2. ساختار منعطف: داده‌های فرم به صورت دینامیک ذخیره و قابل پردازش هستند.

اعتبارسنجی داده‌های فرم

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

const handleSubmit = (event) => {
  event.preventDefault();
  if (!formData.name || !formData.email) {
    alert('لطفاً تمامی فیلدها را پر کنید.');
    return;
  }
  console.log('فرم معتبر است:', formData);
};

نکات:

  • اعتبارسنجی می‌تواند به صورت ساده (مانند چک کردن خالی نبودن فیلدها) یا پیچیده‌تر (مانند بررسی الگوهای ایمیل) انجام شود.
  • برای اعتبارسنجی پیشرفته‌تر، استفاده از کتابخانه‌هایی مثل Formik یا Yup توصیه می‌شود.

ارسال داده‌ها به سرور

برای ارسال داده‌های فرم به یک سرور، از روش‌هایی مانند fetch یا کتابخانه‌های دیگر (مثل Axios) استفاده می‌شود:

const handleSubmit = async (event) => {
  event.preventDefault();
  try {
    const response = await fetch('/api/submit', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify(formData),
    });
    const result = await response.json();
    console.log('نتیجه ارسال:', result);
  } catch (error) {
    console.error('خطا در ارسال:', error);
  }
};

مدیریت فرم‌های پیچیده

برای فرم‌های با ساختار پیچیده‌تر یا تعداد فیلدهای زیاد، استفاده از کتابخانه‌های مدیریت فرم مانند Formik توصیه می‌شود. این کتابخانه با ارائه امکاناتی مانند اعتبارسنجی پیشرفته، مدیریت خودکار state و بهینه‌سازی تعاملات فرم، فرآیند توسعه را ساده‌تر می‌کند.

نکات مهم:

  • حتماً از preventDefault() برای جلوگیری از رفتار پیش‌فرض مرورگر استفاده کنید.
  • داده‌های فرم را قبل از ارسال اعتبارسنجی کنید.
  • برای تجربه بهتر کاربر، می‌توانید از پیام‌های موفقیت یا خطا پس از ارسال فرم استفاده کنید.

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

 

نکات پیشرفته در مدیریت رویدادهای ری اکت

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

مدیریت رویدادها با استفاده از Context

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

مثال:

const EventContext = React.createContext();

function ParentComponent() {
  const handleEvent = () => {
    console.log('رویداد در والد مدیریت شد.');
  };

  return (
    <EventContext.Provider value={handleEvent}>
      <ChildComponent />
    </EventContext.Provider>
  );
}

function ChildComponent() {
  const handleEvent = React.useContext(EventContext);
  return <button onClick={handleEvent}>کلیک کنید</button>;
}

بهینه‌سازی با Debouncing و Throttling

برای مدیریت رویدادهایی که به سرعت پشت سر هم رخ می‌دهند (مانند scroll یا resize)، استفاده از debouncing یا throttling می‌تواند عملکرد را بهبود بخشد. کتابخانه‌هایی مانند lodash ابزارهای آماده‌ای برای این کار ارائه می‌دهند.

مثال:

import { useEffect } from 'react';
import { debounce } from 'lodash';

function App() {
  useEffect(() => {
    const handleScroll = debounce(() => {
      console.log('اسکرول انجام شد');
    }, 300);

    window.addEventListener('scroll', handleScroll);

    return () => window.removeEventListener('scroll', handleScroll);
  }, []);

  return <div>به پایین اسکرول کنید</div>;
}

رویدادهای ترکیبی (Synthetic Events)

ری‌اکت از Synthetic Events استفاده می‌کند که یک لایه انتزاعی بالاتر از رویدادهای بومی مرورگر ارائه می‌دهد. این رویدادها به صورت یکپارچه در تمام مرورگرها عمل می‌کنند. توسعه‌دهندگان می‌توانند با استفاده از متدهای موجود در این رویدادها، اطلاعات دقیقی از نوع رویداد و رفتار آن به دست آورند.

نکته: برای دسترسی به رویداد بومی مرورگر، می‌توانید از event.nativeEvent استفاده کنید:

function handleClick(event) {
  console.log('رویداد ری‌اکت:', event);
  console.log('رویداد بومی:', event.nativeEvent);
}

مدیریت رویدادها در پروژه‌های بزرگ

در پروژه‌های بزرگ، مدیریت مستقیم رویدادها در هر کامپوننت می‌تواند باعث پیچیدگی و افزایش حجم کد شود. استفاده از ابزارهایی مانند Redux برای مدیریت رویدادها و وضعیت برنامه، رویکردی سازمان‌یافته‌تر ارائه می‌دهد.

مثال: مدیریت کلیک در Redux

// تعریف Action
const CLICK_EVENT = 'CLICK_EVENT';
const handleClick = () => ({ type: CLICK_EVENT });

// Reducer
function eventReducer(state = { clicks: 0 }, action) {
  switch (action.type) {
    case CLICK_EVENT:
      return { ...state, clicks: state.clicks + 1 };
    default:
      return state;
  }
}

استفاده از Event Delegation

در برخی موارد، به جای اضافه کردن لیسنر برای هر عنصر، می‌توانید از delegation استفاده کنید. این روش معمولاً برای بهبود عملکرد در لیست‌های طولانی کاربرد دارد.

مثال:

function App() {
  const handleEvent = (event) => {
    if (event.target.tagName === 'BUTTON') {
      console.log('دکمه کلیک شد:', event.target.textContent);
    }
  };

  return (
    <div onClick={handleEvent}>
      <button>دکمه ۱</button>
      <button>دکمه ۲</button>
    </div>
  );
}

تعامل با API‌های بومی مرورگر

برای تعامل با API‌های بومی مانند Drag & Drop یا Pointer Events، می‌توانید لیسنرهای رویدادهای بومی را مستقیماً اضافه کنید:

useEffect(() => {
  const handleDrag = (event) => {
    console.log('آیتم در حال کشیدن است:', event);
  };

  document.addEventListener('drag', handleDrag);

  return () => document.removeEventListener('drag', handleDrag);
}, []);

مدیریت پیشرفته رویدادها در ری‌اکت به توسعه‌دهندگان این امکان را می‌دهد که تعاملات پیچیده را با کارایی و سادگی پیاده‌سازی کنند. با استفاده از تکنیک‌هایی مانند Context، Debouncing، Redux و Event Delegation، می‌توانید کدی ساخت‌یافته‌تر و بهینه‌تر داشته باشید. برای پروژه‌های بزرگ‌تر، ترکیب این روش‌ها با ابزارهای پیشرفته‌تر بهترین راهکار خواهد بود.

جمع بندی:

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

مطالعه بیشتر در رفرنس های خارجی:

Responding to Events (react.dev)

Handling Events (reactjs.org)

React Events (w3schools.com)

لیست کامل مقالات ” ری اکت ” ویکس سِوِن در لینک زیر :

ری اکت چیست؟