پیاده سازی حالت تیره در اپلیکیشن ری اکت – به زبان ساده


امروزه اغلب اپلیکیشنها از یک «حالت تیره» (Dark Mode) استفاده میکنند تا کاربر بتواند بین استفاده از قالب روشن و تیره انتخاب بکند. حالت تیره در طراحی اپلیکیشنها و نرمافزارها رواج فراوانی یافته است. علاوه بر این که «حالت تیره» خود قابلیت جالبی است، میتواند به کاهش مصرف توان دستگاه و همچنین به کاهش فشار بر چشم کاربر نیز کمک کند. در این مقاله با روش پیاده سازی حالت تیره در اپلیکیشن ری اکت آشنا میشویم.
اپلیکیشنی که در این راهنما مورد بررسی قرار میدهیم کدهای CSS فراوانی دارد، اما برای حفظ انسجام صرفاً CSS مرتبط با کارکرد حالت شب/روز را ارائه میکنیم. ابتدا باید کمی با چارچوب مسئله آشنا شویم. اپلیکیشن ریاکت مورد بررسی یک ادیتور متن آنلاین است. کامپوننت App پسزمینه سفیدی دارد و رنگ نوشتهها نیز سیاه است. این قالب رنگی در سراسر اپلیکیشن رعایت شده است.
کد کامپوننت App به صورت زیر است:
چنان که میبینید، کامپوننت App از طریق mapStateToProps به یک prop به نام mode دسترسی دارد. آن را با connect اکسپورت میکنیم. با این که استفاده از ریداکس برای این کارکرد ضرورتی ندارد، اما موجب میشود که اعمال استایلها آسانتر شود و کامپوننتها مدیریت بهتری مییابند. کد reducer به صورت زیر است:
زمانی که CHANGE_MODE فراخوانی میشود، تنها کاری که باید انجام شود، دریافت مقدار mode در حالت reducer و انتساب آن به مقدار مخالف (true/false) کنونی است.
اگر state.mode به صورت true باشد، !state.mode برابر با false خواهد بود و برعکس. در عمل به هیچ payload نیاز نداریم. اکنون با شیوه مدیریت reducer آشنا شدیم، زمان آن رسیده که به بررسی عملی CHANGE_MODE در فایل خود بپردازیم.
از آنجا که هیچ payload وجود ندارد و عملیات toggle در reducer رخ میدهد، تنها کاری که برای تغییر دادن قالب باید انجام دهیم، فراخوانی این تابع در یکی از کامپوننتها است. بدین ترتیب به کامپوننت جدیدی به نام SizeContainer نیاز داریم.
از آنجا که props.mode را از App به SizeContainer میفرستیم، عناصر فرزند در SizeContainer نیز به props.mode دسترسی خواهند داشت. این وضعیت به طور خاص برای کامپوننت فرزندی به نام Buttons اهمیت دارد:
کد فوق دو دکمه زیر را رندر میکند که یکی آیکون لامپ و دیگری آیکون فایل کد است.
چنان که احتمالاً حدس میزنید، دکمه آیکون لامپ حالت تیره را کنترل میکند. در مورد دکمه دوم فعلاً چیزی نمیگوییم. به خط کد زیر توجه کنید:
این تابع changeMode است که از actions اکسپورت میشود. این تابع دکمه آیکون لامپ را به صورت یک رویداد onClick اضافه میکند. روی دکمه کلیک کنید تا اکشن تغییر حالت اجرا شود.
tool tip دکمه لامپ، در صورتی که props.mode برابر با true باشد، عبارت Day Mode را نشان میدهد که معنی آن فعال بودن حالت روز است. همچنین در صورتی که false باشد، عبارت Night Mode را نشان میدهد که معنی آن فعال بودن حالت تیره یا شب است.
اکنون که اپلیکیشن ریاکت خود را تنظیم کردهایم، نوبت بررسی CSS رسیده است.
زمانی که حالت تیره انتخاب نشده باشد، CSS ما کاملاً سرراست خواهد بود. از آنجا که سفید رنگ پیشفرض پسزمینه و سیاه رنگ فونت HTML است، نیازی به ذکر آنها در اینجا وجود ندارد. تنها کاری که باید انجام دهیم، تعیین ارتفاع و عرض App و transition است. با تعیین گذار رنگ پسزمینه و گذار رنگ (هر دو روی 0.3 ثانیه) مطمئن میشویم که گذار از حالت روز به شب روان خواهد بود. در غیر این صورت این گذار بیشتر شبیه به فلاش زدن دوربین عکاسی خواهد بود. اکنون که استایلهای پیشفرض را در اختیار داریم، به طراحی CSS حالت تیره میپردازیم.
در کد فوق استایلهایی برای کلاس دارای نام .night اعلان کردهایم و رنگ پسزمینه و رنگ را صراحتاً طوری مشخص ساختهایم که HTML از مقادیر پیشفرض استفاده نکند. این استایل نیز گذارهایی به صورت زیر دارد:
بدین ترتیب اطمینان مییابیم که وقتی از حالت تیره به حالت روشن جابجا میشویم، گذار همچنان روان خواهد بود. اگر به خاطر داشته باشید، className مربوط به کامپوننت App بر اساس مقدار props.mode تعیین میشود. اگر props.mode به صورت false باشد از className="App" استفاده میکنیم و در غیر این صورت از "className="night استفاده خواهد شد. CSS فوق با تغییر قالب برای کامپوننت App را پیادهسازی میکند، اما برای این که مطمئن شویم استایلهای ضروری بر روی همه عناصر فرزند اعمال خواهد شد، باید چند خط کد دیگر نیز اضافه کنیم:
بدین ترتیب رنگ پسزمینه و رنگ متن برای هر یک از عناصر h1 ،h2 ،p ،div و textarea که درون کامپوننت App قرار دارند عوض خواهد شد. این عناصر بخشهای ادیتور متنی اپلیکیشن textarea و عناوین آن (h2) را شامل میشوند. h1 عنوان خود اپلیکیشن است و تقریباً هر چیز دیگر یا در کانتینر div خود و یا در کانتینر div والدش قرار دارد. با قرار دادن night. در جلوی همه این عناصر، میتوانیم این تگهای فرزند خاص را در زمان تغییر قالب اپلیکیشن هدفگیری کنیم.
همچنین باید اشاره کنیم که در اینجا از سلکتور descendant استفاده کردهایم که به وسیله استفاده از فاصله بین .night و تگ HTML مشخص میشود. بدین ترتیب همه وهلههای فرزند یک عنصر که زیر .night قرار میگیرند، انتخاب خواهند شد. ما از عملگر فرزند (>) استفاده نکردهایم، چون تنها فرزند بیواسطه یک عنصر را انتخاب میکند. همچنین از عملگر همنیای مجاور (+) نیز استفاده نکردیم، زیرا تنها عناصر همنیا را انتخاب میکند. برای مطالعه بیشتر در خصوص انواع سلکتورهای CSS به مقاله زیر مراجعه کنید:
به این ترتیب کار طراحی حالت تیره اپلیکیشن ریاکت تقریباً به پایان رسیده است، اما چند نکته دیگر نیز وجود دارند که باید به آنها توجه داشته باشید. نکته نخست این که چند عنصر مانند دکمههای آبی هستند که لازم نیست رنگ یا رنگ پسزمینه آنها را تغییر دهیم. فرقی نمیکند از حالت تیره یا روشن استفاده کنید، این دکمهها همواره باید آبی با آیکونهای سفید باقی بمانند. برای تضمین این که تغییر className تأثیری روی این دکمهها نمیگذارد، CSS زیر را اضافه میکنیم:
با قرار دادن خط فوق در فایل App.js مطمئن میشویم که CSS قبلی دیگر نمیتواند استایل دکمهها را عوض کند. شاید فکر کنید این CSS باید پس از CSS اصلی بیاید، زیرا طرز کار CSS به صورت آبشاری است و از تقدم عملگرها برای تعیین استایلی که باید روی عنصر اعمال شود بهره میگیرد.
اما در عمل مهم نیست که این خط کد کجا قرار میگیرد. از آنجا که ما از سلکتور کلاس برای هدفگیری دکمهها استفاده میکنیم، استایلهای .night div روی «سطح خصوصیت» (specificity) پایینتر از .night.buttons اعمال نمیشوند. برای کسب اطلاعات بیشتر در این مورد به این مقاله مراجعه کنید:
منطق فوق را میتوان در مورد عنصر آخر نیز مورد استفاده قرار دارد. ما میخواهیم عنصر کانتینر خروجی از حالت تیره استقلال داشته باشد. این بخش دارای نام کلاس.vew است که کد کاربر در آن اجرا میشود و از آنجا که ناحیه خروجی تنها کد نوشته شده از سوی کاربر را نمایش میدهد، باید مطمئن شویم که استایلهای پیشفرض اعمال شدهاند، مگر این که کاربر انواع جدیدی را در بخش CSS وارد کرده باشد.
بقیه خروجی در یک iframe درج میشود که فرزند .view است. این بدان معنی است که استایلهای کاربر کاملاً متمایز از استایلهای اپلیکیشن میمانند. جلوگیری از تغییر رنگ پسزمینه کانتینر خروجی تنها کاری است که به آن نیاز داریم. نتیجه نهایی به صورت زیر است:
بدین ترتیب موفق شدیم حالت تیره را در اپلیکیشن ریاکت خود پیادهسازی کنیم. با چند خط CSS اضافی و کمی منطق props، اپلیکیشن ریاکت ما اکنون به صورت مؤثر و بهینهای مجهز به قابلیت حالت تیره یا شب شده است.
اگر این مطلب برای شما مفید بوده است، آموزشهای زیر نیز به شما پیشنهاد میشوند:
- مجموعه آموزشهای JavaScript (جاوا اسکریپت)
- مجموعه آموزشهای طراحی سایت با HTML و CSS
- آموزش مقدماتی فریمورک React Native برای طراحی نرم افزارهای اندروید و iOS با زبان جاوااسکریپت
- ۲۲ ابزار مهم برای توسعه دهندگان React — فهرست کاربردی
- ۱۰ نکته و ترفند برای بهبود برنامه نویسی React — راهنمای کاربردی
==