কীভাবে একটি অভিযোজিত এবং অ্যাক্সেসযোগ্য টোস্ট উপাদান তৈরি করা যায় তার একটি ভিত্তিগত ওভারভিউ।
এই পোস্টে আমি কীভাবে একটি টোস্ট উপাদান তৈরি করতে হয় সে সম্পর্কে চিন্তাভাবনা ভাগ করতে চাই। ডেমো চেষ্টা করুন.
আপনি যদি ভিডিও পছন্দ করেন তবে এখানে এই পোস্টটির একটি YouTube সংস্করণ রয়েছে:
ওভারভিউ
টোস্টগুলি ব্যবহারকারীদের জন্য অ-ইন্টারেক্টিভ, প্যাসিভ এবং অ্যাসিঙ্ক্রোনাস সংক্ষিপ্ত বার্তা। সাধারণত এগুলি একটি ক্রিয়াকলাপের ফলাফল সম্পর্কে ব্যবহারকারীকে অবহিত করার জন্য একটি ইন্টারফেস প্রতিক্রিয়া প্যাটার্ন হিসাবে ব্যবহৃত হয়।
মিথস্ক্রিয়া
টোস্টগুলি নোটিফিকেশন, সতর্কতা এবং প্রম্পটের বিপরীত কারণ তারা ইন্টারেক্টিভ নয়; তাদের বরখাস্ত করা বা অব্যাহত রাখার জন্য নয়। বিজ্ঞপ্তিগুলি আরও গুরুত্বপূর্ণ তথ্য, সিঙ্ক্রোনাস মেসেজিং যার জন্য ইন্টারঅ্যাকশন প্রয়োজন, বা সিস্টেম স্তরের বার্তাগুলি (পৃষ্ঠা স্তরের বিপরীতে)। অন্যান্য নোটিশ কৌশলগুলির তুলনায় টোস্টগুলি আরও প্যাসিভ।
মার্কআপ
<output>
উপাদানটি টোস্টের জন্য একটি ভাল পছন্দ কারণ এটি স্ক্রিন পাঠকদের জন্য ঘোষণা করা হয়। সঠিক HTML আমাদের জাভাস্ক্রিপ্ট এবং CSS এর সাথে উন্নত করার জন্য একটি নিরাপদ ভিত্তি প্রদান করে এবং সেখানে প্রচুর জাভাস্ক্রিপ্ট থাকবে।
একটি টোস্ট
<output class="gui-toast">Item added to cart</output>
role="status"
যোগ করে এটি আরও অন্তর্ভুক্ত হতে পারে। এটি একটি ফলব্যাক প্রদান করে যদি ব্রাউজার <output>
উপাদানগুলিকে স্পেক অনুযায়ী অন্তর্নিহিত ভূমিকা না দেয়।
<output role="status" class="gui-toast">Item added to cart</output>
একটি টোস্ট পাত্রে
একবারে একাধিক টোস্ট দেখানো যেতে পারে। একাধিক টোস্ট অর্কেস্ট্রেট করার জন্য, একটি ধারক ব্যবহার করা হয়। এই ধারকটি পর্দায় টোস্টগুলির অবস্থানও পরিচালনা করে।
<section class="gui-toast-group">
<output role="status">Wizard Rose added to cart</output>
<output role="status">Self Watering Pot added to cart</output>
</section>
বিন্যাস
আমি ভিউপোর্টের inset-block-end
টোস্টগুলি পিন করতে বেছে নিয়েছি এবং যদি আরও টোস্ট যোগ করা হয়, সেগুলি সেই স্ক্রিন প্রান্ত থেকে স্ট্যাক করে।
GUI ধারক
টোস্টের ধারকটি টোস্ট উপস্থাপনের জন্য সমস্ত লেআউটের কাজ করে। এটি ভিউপোর্টে fixed
এবং কোন প্রান্তে পিন করতে হবে তা নির্দিষ্ট করতে লজিক্যাল প্রপার্টি inset
ব্যবহার করে এবং একই block-end
প্রান্ত থেকে কিছুটা padding
।
.gui-toast-group {
position: fixed;
z-index: 1;
inset-block-end: 0;
inset-inline: 0;
padding-block-end: 5vh;
}
ভিউপোর্টের মধ্যে অবস্থান করার পাশাপাশি, টোস্ট কন্টেইনার হল একটি গ্রিড ধারক যা টোস্টগুলিকে সারিবদ্ধ এবং বিতরণ করতে পারে। আইটেমগুলিকে justify-content
সহ একটি গ্রুপ হিসাবে কেন্দ্রীভূত করা হয় এবং স্বতন্ত্রভাবে justify-items
সাথে কেন্দ্রীভূত হয়। একটু gap
করে ফেলুন যাতে টোস্টগুলি স্পর্শ না করে।
.gui-toast-group {
display: grid;
justify-items: center;
justify-content: center;
gap: 1vh;
}
GUI টোস্ট
একটি পৃথক টোস্টে কিছু padding
, border-radius
সহ কিছু নরম কোণ এবং মোবাইল এবং ডেস্কটপের আকার নির্ধারণে সহায়তা করার জন্য একটি min()
ফাংশন রয়েছে। নিম্নলিখিত CSS-এ প্রতিক্রিয়াশীল আকার টোস্টগুলিকে ভিউপোর্টের 90% বা 25ch
এর চেয়ে বেশি প্রশস্ত হতে বাধা দেয়।
.gui-toast {
max-inline-size: min(25ch, 90vw);
padding-block: .5ch;
padding-inline: 1ch;
border-radius: 3px;
font-size: 1rem;
}
শৈলী
লেআউট এবং পজিশনিং সেটের সাথে, CSS যোগ করুন যা ব্যবহারকারীর সেটিংস এবং ইন্টারঅ্যাকশনের সাথে মানিয়ে নিতে সাহায্য করে।
টোস্ট ধারক
টোস্টগুলি ইন্টারেক্টিভ নয়, তাদের উপর ট্যাপ করা বা সোয়াইপ করা কিছুই করে না, তবে তারা বর্তমানে পয়েন্টার ইভেন্টগুলি গ্রাস করে। নিম্নলিখিত CSS দিয়ে ক্লিক চুরি থেকে toasts প্রতিরোধ করুন.
.gui-toast-group {
pointer-events: none;
}
GUI টোস্ট
কাস্টম বৈশিষ্ট্য, HSL এবং একটি পছন্দ মিডিয়া ক্যোয়ারী সহ টোস্টগুলিকে একটি হালকা বা অন্ধকার অভিযোজিত থিম দিন৷
.gui-toast {
--_bg-lightness: 90%;
color: black;
background: hsl(0 0% var(--_bg-lightness) / 90%);
}
@media (prefers-color-scheme: dark) {
.gui-toast {
color: white;
--_bg-lightness: 20%;
}
}
অ্যানিমেশন
একটি নতুন টোস্ট স্ক্রিনে প্রবেশ করার সাথে সাথে একটি অ্যানিমেশনের সাথে নিজেকে উপস্থাপন করা উচিত। সংক্ষিপ্ত গতির মানগুলি ডিফল্টরূপে translate
মানকে 0
তে সেট করে করা হয়, তবে গতি পছন্দ মিডিয়া ক্যোয়ারীতে গতির মানটিকে দৈর্ঘ্যে আপডেট করে। প্রত্যেকেই কিছু অ্যানিমেশন পায়, কিন্তু শুধুমাত্র কিছু ব্যবহারকারীর টোস্ট দূরত্বে ভ্রমণ করতে পারে।
এখানে টোস্ট অ্যানিমেশনের জন্য ব্যবহৃত কীফ্রেমগুলি রয়েছে। CSS প্রবেশদ্বার, অপেক্ষা এবং টোস্টের প্রস্থান নিয়ন্ত্রণ করবে, সবই এক অ্যানিমেশনে।
@keyframes fade-in {
from { opacity: 0 }
}
@keyframes fade-out {
to { opacity: 0 }
}
@keyframes slide-in {
from { transform: translateY(var(--_travel-distance, 10px)) }
}
টোস্ট উপাদান তারপর ভেরিয়েবল সেট আপ করে এবং কীফ্রেমগুলি অর্কেস্ট্রেট করে।
.gui-toast {
--_duration: 3s;
--_travel-distance: 0;
will-change: transform;
animation:
fade-in .3s ease,
slide-in .3s ease,
fade-out .3s ease var(--_duration);
}
@media (prefers-reduced-motion: no-preference) {
.gui-toast {
--_travel-distance: 5vh;
}
}
জাভাস্ক্রিপ্ট
শৈলী এবং স্ক্রিন রিডার অ্যাক্সেসযোগ্য HTML প্রস্তুত সহ, ব্যবহারকারীর ইভেন্টের উপর ভিত্তি করে টোস্ট তৈরি, সংযোজন এবং ধ্বংস করার জন্য জাভাস্ক্রিপ্টের প্রয়োজন। টোস্ট উপাদানটির বিকাশকারীর অভিজ্ঞতা ন্যূনতম এবং শুরু করা সহজ হওয়া উচিত, যেমন:
import Toast from './toast.js'
Toast('My first toast')
টোস্ট গ্রুপ এবং টোস্ট তৈরি করা
যখন জাভাস্ক্রিপ্ট থেকে টোস্ট মডিউল লোড হয়, তখন এটি একটি টোস্ট কন্টেইনার তৈরি করে পৃষ্ঠায় যোগ করতে হবে। আমি body
এর আগে উপাদানটি যুক্ত করতে বেছে নিয়েছি, এটি z-index
স্ট্যাকিং সমস্যাগুলিকে অসম্ভাব্য করে তুলবে কারণ ধারকটি সমস্ত শরীরের উপাদানগুলির জন্য ধারকটির উপরে থাকে৷
const init = () => {
const node = document.createElement('section')
node.classList.add('gui-toast-group')
document.firstElementChild.insertBefore(node, document.body)
return node
}
init()
ফাংশনটিকে মডিউলে অভ্যন্তরীণভাবে বলা হয়, উপাদানটিকে Toaster
হিসাবে আটকে রাখে:
const Toaster = init()
createToast()
ফাংশন দিয়ে টোস্ট HTML উপাদান তৈরি করা হয়। ফাংশনের টোস্টের জন্য কিছু পাঠ্য প্রয়োজন, একটি <output>
উপাদান তৈরি করে, এটিকে কিছু শ্রেণী এবং বৈশিষ্ট্য দিয়ে সাজায়, পাঠ্য সেট করে এবং নোড প্রদান করে।
const createToast = text => {
const node = document.createElement('output')
node.innerText = text
node.classList.add('gui-toast')
node.setAttribute('role', 'status')
return node
}
এক বা একাধিক টোস্ট পরিচালনা করা
জাভাস্ক্রিপ্ট এখন টোস্ট ধারণ করার জন্য নথিতে একটি ধারক যোগ করে এবং তৈরি টোস্ট যোগ করার জন্য প্রস্তুত। addToast()
ফাংশন এক বা একাধিক টোস্ট পরিচালনা করে। প্রথমে টোস্টের সংখ্যা পরীক্ষা করুন এবং গতি ঠিক আছে কিনা, তারপরে এই তথ্যটি ব্যবহার করে হয় টোস্ট যুক্ত করুন বা কিছু অভিনব অ্যানিমেশন করুন যাতে অন্যান্য টোস্টগুলি নতুন টোস্টের জন্য "ঘর তৈরি করে" বলে মনে হয়।
const addToast = toast => {
const { matches:motionOK } = window.matchMedia(
'(prefers-reduced-motion: no-preference)'
)
Toaster.children.length && motionOK
? flipToast(toast)
: Toaster.appendChild(toast)
}
প্রথম টোস্ট যোগ করার সময়, Toaster.appendChild(toast)
CSS অ্যানিমেশনগুলিকে ট্রিগার করে পৃষ্ঠায় একটি টোস্ট যোগ করে: অ্যানিমেট ইন, ওয়েট 3s
, অ্যানিমেট আউট। flipToast()
বলা হয় যখন বিদ্যমান টোস্ট থাকে, পল লুইস দ্বারা FLIP নামক একটি কৌশল ব্যবহার করে। ধারণাটি হল নতুন টোস্ট যোগ করার আগে এবং পরে পাত্রের অবস্থানের পার্থক্য গণনা করা। টোস্টারটি এখন কোথায় আছে, এটি কোথায় হবে তা চিহ্নিত করার মতো চিন্তা করুন, তারপর এটি যেখানে ছিল সেখান থেকে অ্যানিমেটিং করুন।
const flipToast = toast => {
// FIRST
const first = Toaster.offsetHeight
// add new child to change container size
Toaster.appendChild(toast)
// LAST
const last = Toaster.offsetHeight
// INVERT
const invert = last - first
// PLAY
const animation = Toaster.animate([
{ transform: `translateY(${invert}px)` },
{ transform: 'translateY(0)' }
], {
duration: 150,
easing: 'ease-out',
})
}
CSS গ্রিড লেআউট উত্তোলন করে। যখন একটি নতুন টোস্ট যোগ করা হয়, গ্রিড এটিকে শুরুতে রাখে এবং অন্যদের সাথে ফাঁকা করে। ইতিমধ্যে, একটি ওয়েব অ্যানিমেশন পুরানো অবস্থান থেকে কন্টেইনার অ্যানিমেট করতে ব্যবহৃত হয়।
সব জাভাস্ক্রিপ্ট একসাথে রাখা
যখন Toast('my first toast')
কল করা হয়, একটি টোস্ট তৈরি করা হয়, পৃষ্ঠায় যোগ করা হয় (সম্ভবত নতুন টোস্টের জন্য ধারকটি অ্যানিমেটেড করা হয়), একটি প্রতিশ্রুতি দেওয়া হয় এবং তৈরি টোস্টটি CSS অ্যানিমেশন সমাপ্তির জন্য দেখা হয় ( তিনটি কীফ্রেম অ্যানিমেশন) প্রতিশ্রুতি রেজোলিউশনের জন্য।
const Toast = text => {
let toast = createToast(text)
addToast(toast)
return new Promise(async (resolve, reject) => {
await Promise.allSettled(
toast.getAnimations().map(animation =>
animation.finished
)
)
Toaster.removeChild(toast)
resolve()
})
}
আমি অনুভব করেছি এই কোডের বিভ্রান্তিকর অংশটি Promise.allSettled()
ফাংশন এবং toast.getAnimations()
ম্যাপিং-এ রয়েছে। যেহেতু আমি টোস্টের জন্য একাধিক কীফ্রেম অ্যানিমেশন ব্যবহার করেছি, আত্মবিশ্বাসের সাথে জানার জন্য যে সেগুলি সব শেষ হয়ে গেছে, প্রতিটিকে অবশ্যই JavaScript থেকে অনুরোধ করতে হবে এবং তাদের প্রতিটি finished
প্রতিশ্রুতি সম্পূর্ণ করার জন্য পর্যবেক্ষণ করতে হবে। allSettled
আমাদের জন্য সেই কাজটি করে, এর সমস্ত প্রতিশ্রুতি পূরণ হয়ে গেলে নিজেকে সম্পূর্ণরূপে সমাধান করে। await Promise.allSettled()
ব্যবহার করার অর্থ হল কোডের পরবর্তী লাইনটি আত্মবিশ্বাসের সাথে উপাদানটিকে সরিয়ে দিতে পারে এবং ধরে নিতে পারে টোস্টটি তার জীবনচক্র সম্পূর্ণ করেছে। অবশেষে, কলিং resolve()
উচ্চ স্তরের টোস্ট প্রতিশ্রুতি পূরণ করে যাতে বিকাশকারীরা টোস্ট দেখানোর পরে পরিষ্কার করতে বা অন্য কাজ করতে পারে।
export default Toast
সবশেষে, Toast
ফাংশনটি মডিউল থেকে রপ্তানি করা হয়, অন্যান্য স্ক্রিপ্ট আমদানি ও ব্যবহার করার জন্য।
টোস্ট উপাদান ব্যবহার করে
টোস্ট ব্যবহার করে, বা টোস্টের বিকাশকারীর অভিজ্ঞতা, Toast
ফাংশন আমদানি করে এবং এটিকে একটি বার্তা স্ট্রিং দিয়ে কল করে করা হয়।
import Toast from './toast.js'
Toast('Wizard Rose added to cart')
ডেভেলপার যদি টোস্ট দেখানোর পরে পরিষ্কার করার কাজ বা যাই হোক না কেন, তারা অ্যাসিঙ্ক ব্যবহার করতে পারে এবং অপেক্ষা করতে পারে ।
import Toast from './toast.js'
async function example() {
await Toast('Wizard Rose added to cart')
console.log('toast finished')
}
উপসংহার
এখন যেহেতু আপনি জানেন যে আমি কীভাবে এটি করেছি, আপনি কেমন হবে‽ 🙂৷
আসুন আমাদের পদ্ধতির বৈচিত্র্য আনুন এবং ওয়েবে তৈরি করার সমস্ত উপায় শিখি। একটি ডেমো তৈরি করুন, আমাকে লিঙ্কগুলি টুইট করুন এবং আমি নীচের সম্প্রদায়ের রিমিক্স বিভাগে এটি যুক্ত করব!
কমিউনিটি রিমিক্স
- HTML/CSS/JS সহ @_developit : ডেমো এবং কোড
- HTML/CSS/JS সহ Joost van der Schee : ডেমো এবং কোড