Event Delegation Pattern বিস্তারিত বাংলায়
Event Delegation Pattern হলো একটি জাভাস্ক্রিপ্ট ডিজাইন প্যাটার্ন, যা ইভেন্ট হ্যান্ডলিংকে দক্ষ ও স্কেলেবল করে তোলে। এর মূল ধারণা হলো:
“প্যারেন্ট এলিমেন্টে একটি মাত্র ইভেন্ট লিসেনার ব্যবহার করে সব চাইল্ড এলিমেন্টের ইভেন্ট ম্যানেজ করা।”
কেন ব্যবহার করা হয়?
- পারফরম্যান্স অপ্টিমাইজেশন:
প্রতিটি চাইল্ড এলিমেন্টে আলাদা ইভেন্ট লিসেনার যোগ করলে মেমরি বেশি খরচ হয়। একটি প্যারেন্টে একটি লিসেনার দিয়েই সব চাইল্ড কন্ট্রোল করা যায়। -
ডাইনামিক এলিমেন্ট সাপোর্ট:
পরবর্তীতে যোগ হওয়া নতুন চাইল্ড এলিমেন্টগুলোও অটোমেটিক্যালি ইভেন্ট হ্যান্ডল করতে পারে। আলাদা করে নতুন লিসেনার যোগ করার দরকার নেই। - কোড সিমপ্লিসিটি:
কম কোডে বেশি কাজ করা যায়। মেইনটেইন করা সহজ হয়।
কিভাবে কাজ করে?
এটি Event Bubbling কনসেপ্টের উপর ভিত্তি করে কাজ করে।
- Event Bubbling: যখন কোনো এলিমেন্টে ইভেন্ট (যেমন
click) ঘটে, তখন সেই ইভেন্টটি DOM ট্রি অনুসারে উপরের দিকে প্যারেন্ট এলিমেন্টে পৌঁছায় (বাবল আপ হয়)। - প্যারেন্ট এলিমেন্টে থাকা ইভেন্ট লিসেনার সেই ইভেন্টটি ক্যাচ করে এবং
event.targetপ্রপার্টি ব্যবহার করে বুঝতে পারে কোন চাইল্ড এলিমেন্ট থেকে ইভেন্টটি এসেছে।
উদাহরণ: টু-ডু লিস্ট
ধরুন, আমাদের একটি টু-ডু লিস্ট আছে যেখানে প্রতিটি আইটেম ক্লিক করলে সেটি ডিলিট হবে।
HTML স্ট্রাকচার:
<ul id="todo-list">
<li>Task 1</li>
<li>Task 2</li>
<li>Task 3</li>
</ul>
সাধারণ পদ্ধতি (ইভেন্ট ডেলিগেশন ছাড়া):
// প্রতিটি li-তে আলাদা লিসেনার যোগ করা
document.querySelectorAll("#todo-list li").forEach((item) => {
item.addEventListener("click", function () {
this.remove(); // ক্লিক করা li ডিলিট হবে
});
});
সমস্যা:
- নতুন আইটেম যোগ হলে সেগুলোর জন্য আলাদাভাবে লিসেনার যোগ করতে হবে।
- ১০০টি আইটেম থাকলে ১০০টি লিসেনার তৈরি হবে (মেমরি ওয়েস্ট)।
Event Delegation ব্যবহার করে:
// শুধু প্যারেন্ট ul-এ একটি লিসেনার যোগ করা
document
.getElementById("todo-list")
.addEventListener("click", function (event) {
// চেক করা হচ্ছে ক্লিক করা এলিমেন্ট li কিনা
if (event.target.tagName === "LI") {
event.target.remove(); // ক্লিক করা li ডিলিট হবে
}
});
ব্যাখ্যা:
event.targetবলে দেয় কোন এলিমেন্টে ক্লিক হয়েছে (যেমনli)।- প্যারেন্ট
ul-এ একটি মাত্র লিসেনার থাকলেও সবli-এর ক্লিক হ্যান্ডেল হচ্ছে। - নতুন
liযোগ হলেও কোনো অতিরিক্ত কোড লাগবে না।
কোডের গুরুত্বপূর্ণ অংশ
parentElement.addEventListener("click", function (event) {
// চেক: ইভেন্ট টার্গেট কাঙ্ক্ষিত এলিমেন্ট কিনা
if (event.target.matches(".child-selector")) {
// কাজ করো
}
});
প্রয়োজনীয় চেকগুলো:
event.target: যে এলিমেন্টে ইভেন্ট ঘটেছে।event.currentTarget: যে এলিমেন্টে লিসেনার অ্যাটাচ করা হয়েছে (প্যারেন্ট)।- সিলেক্টর চেক:
event.target.matches('selector')দিয়ে যাচাই করা যায় টার্গেট কাঙ্ক্ষিত এলিমেন্ট কিনা।
সুবিধা সমূহ
| সুবিধা | ব্যাখ্যা |
|---|---|
| মেমরি সাশ্রয় | একটি লিসেনার দিয়ে অনেক এলিমেন্ট ম্যানেজ করা যায়। |
| ডাইনামিক কন্টেন্ট | নতুন এলিমেন্ট যোগ/সরানোর পরেও কোড আপডেট করার দরকার নেই। |
| কোড রিডাকশন | কম কোডে বেশি কাজ হয়, বাগ কম হয়। |
| পারফরম্যান্স | লিসেনার সংখ্যা কম থাকায় পেজ লোড ফাস্ট হয়। |
ব্যবহারের ক্ষেত্র
- লিস্ট/টেবিল: যেখানে অনেক রো আছে (যেমন শপিং কার্ট)।
- বাটন গ্রুপ: একাধিক বাটনে একই ধরনের অ্যাকশন।
- ডাইনামিক ইউআই: যেখানে এলিমেন্ট রানটাইমে যোগ/সরানো হয় (যেমন চ্যাট অ্যাপ)।
- মোবাইল অ্যাপ: টাচ ইভেন্ট অপ্টিমাইজেশনের জন্য।
সতর্কতা
- ইভেন্ট টার্গেট ভুল হতে পারে: যদি চাইল্ড এলিমেন্টের ভেতর আরও নেস্টেড এলিমেন্ট থাকে (যেমন
<li><span>Text</span></li>), তাহলেevent.targetspanহতে পারে। এক্ষেত্রেclosest()মেথড ব্যবহার করুন:if (event.target.closest("li")) { event.target.closest("li").remove(); } - ইভেন্ট স্টপ প্রোপাগেশন: যদি কোনো চাইল্ড এলিমেন্টে
event.stopPropagation()ব্যবহার করা হয়, তাহলে ইভেন্ট প্যারেন্ট পর্যন্ত পৌঁছাবে না।
সীমাবদ্ধতা:
-
টাইমিং সমস্যা: কখনো কখনো, যদি কোনো ইভেন্ট খুব দ্রুত ঘটতে থাকে (যেমন মাউস মুভ বা টাইপিং), তাহলে ইভেন্ট ডেলিগেশন প্যাটার্নের কারণে সমস্যাও হতে পারে। তবে এটি বেশিরভাগ সাধারণ ব্যবহারের জন্য কার্যকর।
-
ইভেন্টের ধরন: কিছু ইভেন্ট যেমন
focusবাblur-এ ডেলিগেশন কাজ করতে পারে না, কারণ এ ধরনের ইভেন্টগুলি বুব্বলিং প্রোপাগেশন অনুসরণ করে না।
সারসংক্ষেপ
Event Delegation Pattern হলো “এক লিসেনার, সব চাইল্ড কন্ট্রোল” এর একটি স্মার্ট কৌশল। এটি মেমরি বাঁচায়, কোড সহজ করে এবং ডাইনামিক কন্টেন্ট হ্যান্ডেল করতে সাহায্য করে। বড় অ্যাপ্লিকেশনে এটি অত্যন্ত কার্যকরী! 🚀