تتبع تغييرات الحالة باستخدام إضافة Zustand
يُعد Zustand بديلاً سريعاً للغاية وبسيطاً لحلول إدارة الحالة مثل Redux. باستخدام إضافة Zustand الخاصة بنا، يمكنك تتبع تغييرات الحالة طوال دورة حياة تطبيقك.
تثبيت الإضافة
Section titled تثبيت الإضافةيمكن تثبيت الإضافة بسهولة باستخدام الأمر التالي:
npm i @openreplay/tracker-zustand
استخدام الإضافة في تطبيق Next.js
Section titled استخدام الإضافة في تطبيق Next.jsملاحظة: يوجد الكود المصدري الكامل لهذا المثال في مستودع Github هذا، فألقِ نظرة عليه إذا كنت بحاجة إلى مزيد من التفاصيل.
يتيح موفّر Context الذي أستخدمه تكوين Tracker Plugins، لذلك بعد تثبيت الإضافة، سنقوم بتعديل ملف _app.tsx بإضافة عبارة الاستيراد أدناه:
import trackerZustand from '@openreplay/tracker-zustand'
وبعد ذلك، يجب أن تبدو دالة App الرئيسية على النحو التالي:
export default function MyApp({ Component, pageProps }: AppProps) {
const Layout = (Component as any).Layout || Noop
useEffect(() => {
document.body.classList?.remove('loading')
}, [])
let plugins = [
{
fn: trackerZustand,
name: 'zustand',
},
]
return (
<TrackerProvider
config={{
plugins
}}
>
<Head />
<ManagedUIContext>
<Layout pageProps={pageProps}>
<Component {...pageProps} />
</Layout>
</ManagedUIContext>
</TrackerProvider>
)
}
سيؤدي هذا تلقائياً إلى تهيئة الإضافة مع المتتبّع (tracker)، كما سيتيح أيضاً البيانات المُعادة من تنفيذ الدالة الرئيسية للإضافة.
الخطوة التالية هي إنشاء store، وسنقوم بعد قليل بإضافة منتجات من موقع التجارة الإلكترونية إلى هذا الـ store.
إنشاء store باستخدام إضافة Zustand
Section titled إنشاء store باستخدام إضافة Zustandيتطلب استخدام الإضافة منك تغيير الطريقة التي تنشئ بها عادةً نسخة store في Zustand، لذا لا تتردد في نسخ الكود التالي ولصقه كقالب:
import create from 'zustand'
const productStore = null
export const useProductsStore = (logger) => {
if (!productStore) {
//use a singleton pattern to make sure we only create the store the first time around
productStore = createProductStore(logger)
}
return productStore
}
function createProductStore(logger) {
return create(
logger((set) => ({
products: [],
addProduct: (p) =>
set((state) => {
const prodFound = state.products.find((prod) => {
return p.id == prod.id
})
if (prodFound) {
p.amount = prodFound.amount + 1
} else {
p.amount = 1
}
return {
products: [...state.products.filter((prod) => p.id != prod.id), p],
}
}),
}))
)
}
ستقوم الدالة المُصدَّرة بإنشاء الـ store في المرة الأولى التي تُستخدم فيها، لكنها بعد ذلك ستعيد ببساطة الـ store الموجود مسبقاً.
من ناحية أخرى، ستقوم دالة createProductStore بإنشاء الـ store، وتعيين المنطق الخاص بطريقة addProduct، واستخدام دالة logger التي نوفّرها لها. وكما خمّنت على الأرجح، فإن هذا الـ logger هو ذاته الذي حصلنا عليه من استخدام إضافة Zustand.
وبعد أن أصبح الـ store جاهزاً، يمكننا البدء بإضافة المنتجات إليه.
إضافة المنتجات إلى الـ store الخاص بنا
Section titled إضافة المنتجات إلى الـ store الخاص بنايُصدِّر الـ store طريقة addProduct، التي تتولى تتبع المنتجات في سلة التسوق لدينا.
لاستيراد الدالة إلى الكود الخاص بك، عليك القيام بما يلي:
- استيراد موفّر السياق (context provider) الخاص بالمتتبّع واستخدامه.
- الحصول على السمة pluginsReturnedValue من السياق.
- استخدامها لاستدعاء طريقة zustand (لاحظ أن الطريقة تحمل اسم السمة name عند تكوين الإضافة أعلاه)، وتمرير سمة من نوع string لإعطاء الـ store اسماً (سيُستخدَم هذا الاسم أثناء إعادة التشغيل).
- ستُعيد طريقة zustand الـ logger الذي سنستخدمه عند استدعاء الـ hook المسمى useProductStore من zustand (الذي أنشأناه للـ store).
يبدو هذا كله كثيراً، لكنه في جوهره يتلخص في الآتي:
const { pluginsReturnedValues } = useContext(TrackerContext)
const useProductStoreTracked = useProductsStore(
pluginsReturnedValues.zustand('products_store')
)
const { addProduct } = useProductStoreTracked()
الآن، يمكنك ببساطة استدعاء طريقة addProduct، وستقوم بإضافة منتجات السلة إلى الـ store.
قراءة البيانات من الـ store
Section titled قراءة البيانات من الـ storeكما يمكنك أن تتوقع على الأرجح، إذا أردت قراءة البيانات من store الخاص بـ Zustand، فعليك اتباع المجموعة نفسها من الخطوات. لكن بدلاً من أخذ طريقة addProduct، سنقوم بتفكيك مصفوفة products.
هكذا:
const { pluginsReturnedValues } = useContext(TrackerContext)
const useProductStoreTracked = useProductsStore(
pluginsReturnedValues.zustand('products_store')
)
const { products } = useProductStoreTracked()
بعد ذلك يمكنك التكرار على المنتجات بشكل اعتيادي:
<div>
{products.map((p: Product, idx: Number) => {
return (
<div key={+idx}>
{p.name}({p.amount})
</div>
)
})}
</div>
يقوم الكود أعلاه بالتكرار على قائمة المنتجات وعرضها على الشاشة. وكلما أضفت منتجاً آخر إلى الـ store، سيتم تحديثه هنا أيضاً.
مزامنة الحالة مع session replay
Section titled مزامنة الحالة مع session replayلمراجعة تغييرات الحالة أثناء إعادة التشغيل، ما عليك سوى النقر على علامة التبويب الجديدة Zustand في الزاوية السفلية اليمنى:

عند النقر عليها، سترى محتوى الـ store، وستتمكن من فهم كيفية تغيره والبيانات المحفوظة بداخله:

لاحظ اسم الـ store، فهو في الواقع الاسم الذي استخدمناه عند استدعاء طريقة zustand لأول مرة.
لديك أسئلة؟
Section titled لديك أسئلة؟إذا كنت لا تستخدم Next.js بل إطار عمل آخر، فمن المفترض أن تكون طريقة الاستخدام مشابهة، والمفتاح هو الطريقة التي تنشئ بها الـ store. وما دمت تجد طريقة لاستخدام دالة logger عند إنشائه، فستتمكن من تتبع الحالة أثناء إعادة التشغيل.
يمكنك الاطلاع على هذا المستودع للحصول على الكود المصدري الكامل لتطبيق Next.js يعمل باستخدام إضافة Zustand.
إذا واجهت أي مشكلات في إعداد إضافة Zustand في مشروعك، فيرجى التواصل معنا عبر مجتمع Slack الخاص بنا وطرح أسئلتك مباشرةً على مطوّرينا!