مرجع إلى الواجهة البرمجية للخطافات في React
useContext
الخطافات هي إضافة جديدة إلى الإصدار 16.8 في React، إذ تسمح لك باستعمال ميزة الحالة وميزات React الأخرى دون كتابة أي صنف.
تشرح هذه الصفحة الواجهات البرمجية للخطافات المضمَّنة في React.
إن كان موضوع الخطافات جديدًا بالنسبة لك، فيرجى الرجوع إلى صفحة "مدخل إلى الخطافات" وقراءتها أولًا. قد تجد أيضًا الكثير من المعلومات المفيدة في قسم الأسئلة الشائعة.
الخطافات الأساسية
useState
const [state, setState] = useState(initialState);
يعيد هذا الخطاف قيمةً ذات حالة، ودالةً لتحديث هذه القيمة.
أثناء عملية التصيير الأولية، الحالة المعادة (state
) هي نفسها القيمة المُمرَّرة كأول وسيط (initialState
).
تُستعمَل الدالة setState
لتحديث الحالة، إذ تقبل قيمةً جديدةً للحالة وتدرج في الطابور عملية إعادة تصيير لمكون.
setState(newState);
أثناء عمليات إعادة التصيير اللاحقة، القيمة الأولى التي يعيدها الخطاف useState
ستبقى دومًا أحدث حالة بعد تطبيق التحديثات.
ملاحظة: تضمن React أن هوية الدالة setState
مستقرة ولن تتغير عند إعادة التصيير. لهذا فأنه من الآمن حذف قائمة الاعتمادات الخاصة بـ useEffect
أو useCallback
.
تحديثات عبر تمرير دالة
إن حُسبَت الحالة الجديدة باستعمال الحالة السابقة، فيمكنك تمرير دالة إلى setState
. ستستقبل الدالة القيمة السابقة، وتعيد القيمة المحدَّثة. إليك مثالٌ عن مكون عداد يستعمل كلا الشكلين للخطاف setState
:
function Counter({initialCount}) {
const [count, setCount] = useState(initialCount);
return (
<>
Count: {count}
<button onClick={() => setCount(initialCount)}>Reset</button>
<button onClick={() => setCount(prevCount => prevCount - 1)}>-</button>
<button onClick={() => setCount(prevCount => prevCount + 1)}>+</button>
</>
);
}
يستعمل الزر "+" والزر "-" الشكل الدالِّي (functional form) لأنَّ القيمة المحدَّثة تعتمد على القيمة السابقة. ولكن الزر "Reset" يستعمل الشكل الاعتيادي لأنَّه يضبط العداد إلى القيمة 0 دومًا.
إن أعادت دالة التحديث قيمة مساوية لقيمة الحالة الراهنة ، فسيُتخطى التصيير التالي تخطيا كاملًا.
ملاحظة: خلافًا للتابع setState
في مكونات الأصناف، الخطاف useState
لا يدمج كائنات التحديث (update objects). يمكنك تكرار هذا السلوك عبر دمج شكل الدالة المحدِّثة مع معامل النشر للكائن:
setState(prevState => {
// Object.assign would also work
return {...prevState, ...updatedValues};
});
هنالك خيار آخر وهو استعمال الخطاف useReducer
الذي يعدُّ مناسبًا لإدارة كائنات حالة تحوي عدة قيم فرعية.
الحالة الأولية الكسولة
الوسيط initialState
هو الحالة المستعملة أثناء عملية التصيير الأولى (initial render). في عمليات التصيير اللاحقة، سيُهمَل هذا الوسيط. إن كانت الحالة الأولية هي ناتج عملية حساب معقدة تؤثر على الأداء، فيمكنك أن تمرِّر دالةً تُنفَّذ مرةً واحدةً في أول عملية تصيير عوضًا عن ذلك:
const [state, setState] = useState(() => {
const initialState = someExpensiveComputation(props);
return initialState;
});
عدم تغير الحالة عن إجراء تحديث عليها
إن حدَّث خطاف حالة وكانت القيمة المحدَّثة نفسَ قيمة الحالة الحالية، فلن تتكبد React عناء تصيير الابن أو تنفيذ التأثيرات. (تستعمل React الخوارزمية Object.is لإجراء عملية الموازنة.)
useEffect
useEffect(didUpdate);
يقبل هذا الخطاف دالةً تحوي أمرًا يكون غالبًا شيفرةً ذات تأثير.
التعديلات، والاشتراكات، والمؤقتات، والسجلات، والتسجيل (logging)، والتأثيرات الجانبية الأخرى غير مسموحٍ بها داخل الجسم الرئيسي لمكون دالة (يشار إليه على أنَّه مرحلة تصيير React [أي render phase]). سيؤدي فعل ذلك إلى حصول أخطاءٍ مربكة مع تناقضات في واجهة المستخدم.
عوضًا عن ذلك، استعمل الخطاف useEffect
. الدالة المُمرَّر إليه ستُنفَّذ بعد الانتهاء من التصيير على الشاشة. فكر في التأثيرات وكأنَّها مخرج هروب (escape hatch) من عالم React الوظيفي البحت إلى العالم الأمري.
افتراضيًّا، تُنفَّذ التأثيرات بعد كل كل عملية تصيير مكتملة، ولكن يمكنك اختيار تنفيذها فقط عند تغير قيم محدَّدة.
تنظيف تأثير
تنشئ التأثيرات غالبًا موارد تحتاج للتنظيف قبل أن يغادر المكون الشاشة مثل معرِّف اشتراك أو مُؤقِّت. لفعل ذلك، قد تعيد الدالة المُمرَّرة إلى الخطاف useEffect
دالةً تجري عملية التنظيف. على سبيل المثال، يمكن إنشاء اشتراك بالشكل التالي:
useEffect(() => {
const subscription = props.source.subscribe();
return () => {
// تنظيف الاشتراك
subscription.unsubscribe();
};
});
تُنفَّذ دالة التنظيف قبل حذف المكون من واجهة المستخدم لمنع حدوث تسريب في الذاكرة. أضف إلى ذلك أنَّه إن صُيَّر مكوِّنٌ عدَّة مرات (كما تفعل عادةً)، فسيُنظَّف التأثير السابق قبل تنفيذ التأثير اللاحق. في مثالنا، هذا يعني أنَّه يُنشَأ اشتراك جديد في كل تحديث. لتجنب تنفيذ التأثير عند كل عملية تحديث، ارجع إلى القسم التالي.
توقيت التأثيرات
خلافًا للتابعين componentDidMount
و componentDidUpdate
، تُنفَّذ الدالة المُمرَّرة إلى الخطاف useEffect
بعد التخطيط والرسم أثناء حدث مؤجَّل (deferred event). هذا يجعلها مناسبةً للاستعمال مع العديد من التأثيرات الجانبية الشائعة مثل ضبط الاشتراكات ومعالجات الحدث لأنَّ أغلب أنواع العمل لا يجب أن يحجز المتصفح عن تحديث الشاشة.
على أية حال، لا يمكن تأجيل جميع التأثيرات. على سبيل المثال، يجب أن تُنفَّذ التعديلات التي تجرَى على DOM والظاهرة للمستخدم بشكل متزامن قبل تنفيذ عملية الرسم التالية، لذا لا يلاحظ المستخدم أية تناقضات بصرية. (الفارق من ناحية النظرية مشابهٌ للفارق بين مستمعي حدث نشطين مقابل مستمعي حدث خاملين.) توفر React من أجل هذه الأنواع من التأثيرات خطافًا إضافيًّا يدعى useLayoutEffect
. يملك هذا الخطاف نفس التوقيع الذي يملكه الخطاف useEffect
. الاختلاف الوحيد بينهما هو وقت التنفيذ.
رغم أنَّ الخطاف useEffect
مؤجَّل لبعد انتهاء المتصفح من الرسم، فإنَّ تنفيذه قبل أية عمليات تصيير أخرى أمرٌ مؤكد الحدوث، إذ تنفذ React أية تأثيرات لتصيير سابق قبل بدء تحديث جديد.
تنفيذ تأثير شرطيًّا
السلوك الافتراضي للتأثيرات هو تنفيذ التصيير بعد كل عملية تصيير مكتملة. بهذه الطريقة، يعاد إنشاء تأثيرٍ دومًا إن تغيرت إحدى مدخلاته.
على أية حال، هذا السلوك قد يبدو مبالغًا فيه بشدة في بعض الحالات كما في حالة مثال الاشتراك من القسم السابق. لا نحتاج إلى إنشاء اشتراك جديد في كل تحديث إلا إذا تغيرت الخاصية source
.
لتنفيذ ذلك، مرِّر وسيطًا ثانيًّا إلى الخطاف useEffect
يمثِّل مصفوفة القيم التي يعتمد عليها التأثير. يبدو الآن مثالنا المعدَّل بالشكل التالي:
useEffect(
() => {
const subscription = props.source.subscribe();
return () => {
subscription.unsubscribe();
};
},
[props.source],
);
الآن، سيعاد إنشاء الاشتراك عند تغيِّر props.source
.
تمرير مصفوفة فارغة []
من المدخلات يخبر React أنَّ تأثيراتك لا تعتمد على أية قيم من المكونات؛ لذلك، سيُنفَّذ ذلك التأثير عند الوصل (mount) ويُنظَّف عند الفصل (unmount) ولن تُنفَّذ عند التحديثات.
ملاحظة: إذا أردت استخدام هذا التحسين ، فتأكد من أن المصفوفة تتضمن جميع القيم من نطاق المكوّن (مثل الخاصيات والحالة) التي تتغير بمرور الوقت، والتي يستخدمها التأثير. خلا ذلك، ستشير شيفرتك البرمجيةإلى قيم لا معنى لها من مُصيِّرات سابقة. تعلم كيف تتعامل مع الدوال، وكيف تتصرف عندما تتغير قيم المصفوفة كثيرًا.
إذا أردت تنفيذ تأثير ثمّ تنظيفَه مرة واحدة فقط ( mount
وunmount
) ، يمكنك تمرير مصفوفة فارغة ([]
) كوسيط ثانية. هذا يخبر React أن تأثيرك لا يعتمد على أي قيمة من قيم الخاصيات (props) أو الحالة (state) ، لذلك لا حاجة إلى إعادة التنفيذ. يتم التعامل مع هذا كحالة خاصة - إذ تتبع ألكيفية التي تُعالج بها مصفوفة التبعيات.
إذا مرّرت مصفوفة فارغة ([]
) ، فستحافظ الخاصيات والحالة داخل التأثير دائمًا على قيمها الأولية. صحيح أنّ تمرير []
كوسيط ثان أقرب إلى النموذج المألوف في componentDidMount
و componentWillUnmount
، إلا أنّ هناك عادةً حلولا أفضل لتجنب إعادة تنفيذ التأثيرات بكثرة. لا تنسَ أيضًا أن React ترجئ تنفيذ useEffect
حتى يكتمل رسم المتصفح، لذا فإن القيام بعمل إضافي لا يمثل مشكلة.
نوصي باستخدام القاعدة الشاملة كجزء من حزمة eslint-plugin-reaction-hooks. إذ أنها تطلق تحذيرا في حال لم تُحدد التبعيات على الوجه الصحيح.
مصفوفة المدخلات لا تُمرَّر كوسائط إلى دالة التأثير. نظريًّا، إليك ما الذي تمثله: كل قيمة أشير إليها داخل دالة التأثير يجب أن تظهر أيضًا في مصفوفة المدخلات. في المستقبل، قد يصبح المصرِّف متقدمًا بما فيه الكفاية لإنشاء هذه المصفوفة تلقائيًا.
useContext
const context = useContext(Context);
يقبل هذا الخطاف كائن سياق (context object، أي القيمة المعادة من React.createContext
) ويعيد قيمة السياق الحالي كما أُعطيَت من قبل الخاصية value
لأقرب موفِّر سياق (context provider) فوق المكون المُستدعي في الشجر.
عندما يجري تحديث السياق، سيُطلِق (trigger) هذا الخطاف عملية تصيير مع أحدث قيمة value
للسياق. حتى لو استخدم أحد العناصر السالفة React.memo أو shouldComponentUpdate
، فستستمر إعادة التصيير بدءًا من المكون نفسه باستخدام useContext
.
لا تنس أنه ينبغي أن يكون وسيط useContext
هو كائن السياق نفسه:
- صحيح:
useContext (MyContext)
- غير صحيح:
useContext (MyContext.Consumer)
- غير صحيح:
useContext (MyContext.Provider)
المكون المُستدعي useContext
سيعيد دائمًا التصيير كلما تغيرت قيمة السياق. إذا كانت إعادة تصيير المكون مكلفة، فيمكنك تحسين الأداء باستخدام الذاكرة memoization.
ملاحظة: إذا كنت معتادًا على الواجهة البرمجية الخاصة بالسياق قبل الخطافات، فإن useContext (MyContext)
تكافئ static contextType = MyContext
في الصنف، أو تكافئ <MyContext.Consumer>
.
تتيح لك useContext (MyContext)
قراءة السياق والاشتراك في تغييراته وحسب. ما زلت بحاجة إلى <MyContext.Provider>
أعلاه في الشجرة لتوفير قيمة لهذا السياق.
مثال جامع:
const themes = {
light: {
foreground: "#000000",
background: "#eeeeee"
},
dark: {
foreground: "#ffffff",
background: "#222222"
}
};
const ThemeContext = React.createContext(themes.light);
function App() {
return (
<ThemeContext.Provider value={themes.dark}>
<Toolbar />
</ThemeContext.Provider>
);
}
function Toolbar(props) {
return (
<div>
<ThemedButton />
</div>
);
}
function ThemedButton() {
const theme = useContext(ThemeContext);
return (
<button style={{ background: theme.background, color: theme.foreground }}>
I am styled by theme context!
</button>
);
}
تم تعديل هذا المثال من مثال سابق في توثيق السياق لتلائم الخطافات، حيث يمكنك العثور على مزيد من المعلومات حول كيفية استخدام السياق.
خطافات إضافية
الخطافات التالية هي إمَّا شكل آخر للخطافات الأساسية أو يُلجَأ إليها في حالات محدَّدة فقط. لا تجهد نفسك بتعلمهم الآن إن لم تكن بحاجة لهم.
useReducer
const [state, dispatch] = useReducer(reducer, initialArg, init);
هذا الخطاف هو بديل للخطاف useState
. يقبل هذا الخطاف مخفِّضًا (reducer) من النوع (state, action) => newState
ويعيد الحالة الحالية مقرونةً مع التابع dispatch
. (إن كانت المكتبة Redux مألوفةً لك، فأنت تعرف مسبقًا كيف يعمل ذلك.)
يفضَّل استعمال الخطاف useReducer
عن الخطاف useState
عندما يكون هنالك شيفرة حالة معقدة تتضمن قيم فرعية متعددة أو عندما تعتمد الحالة التالية على سابقتها. الخطاف useReducer
يمكِّنك أيضًا من تحسين الأداء للمكونات التي تستدعي تحديثات عميقة لأنَّه يمكِّنك من تمرير التابع dispatch
للداخل بدلًا من ردود النداء.
إليك مثال العداد الذي أعيد كتابته من القسم useState
ليستعمل مخفِّضًا:
const initialState = {count: 0};
function reducer(state, action) {
switch (action.type) {
case 'increment':
return {count: state.count + 1};
case 'decrement':
return {count: state.count - 1};
default:
throw new Error();
}
}
function Counter() {
const [state, dispatch] = useReducer(reducer, initialState);
return (
<>
Count: {state.count}
<button onClick={() => dispatch({type: 'decrement'})}>-</button>
<button onClick={() => dispatch({type: 'increment'})}>+</button>
</>
);
}
ملاحظة: يضمن React أن تبقى هوية الدالة dispatch
ثابتة لا تتغير عند إعادة التصيير. لهذا فأنه من الآمن حذف قائمة التبعية الخاصة بـ useEffect
أو useCallback
.
تحديد الحالة الأولية
هنالك طريقتان مختلفان لتهيئة حالة الخطاف useReducer
يمكنك الاختيار بينهما بناءً على الحالة المستعملة آنذاك. الطريقة الأبسط هي تمرير الحالة الأولية كوسيطٍ ثانٍ:
const [state, dispatch] = useReducer(
reducer,
{count: initialCount}
);
ملاحظة: لا تستعمل React الوسيط state = initialState
المتعارف عليه والشائع في المكتبة Redux. القيمة الأولية تعتمد أحيانًا على خاصيات وبذلك تُحدَّد من استدعاء الخطاف. إن كنت تشعر بشدة حيال ذلك، فيمكنك استدعاء useReducer(reducer, undefined, reducer)
لمحاكاة سلوك Redux ولكن لا نشجع على ذلك.
التهيئة الكسولة
يمكنك أيضًا إنشاء الحالة الأولية بتكاسل (lazily) عبر تمرير الدالة init
كوسيط ثالث. ستُضبَط الحالة الأولية إلى init(initialArg)
.
يمكِّنك ذلك من استخراج الشيفرة لحساب الحالة الأولية خارج المُخفِّض (reducer). هذا الأمر مفيدٌ وعملي لإعادة ضبط الحالة لاحقًا بالاستجابة لفعلٍ ما.
function init(initialCount) {
return {count: initialCount};
}
function reducer(state, action) {
switch (action.type) {
case 'increment':
return {count: state.count + 1};
case 'decrement':
return {count: state.count - 1};
case 'reset':
return init(action.payload);
default:
throw new Error();
}
}
function Counter({initialCount}) {
const [state, dispatch] = useReducer(reducer, initialCount, init);
return (
<>
Count: {state.count}
<button
onClick={() => dispatch({type: 'reset', payload: initialCount})}>
Reset
</button>
<button onClick={() => dispatch({type: 'decrement'})}>-</button>
<button onClick={() => dispatch({type: 'increment'})}>+</button>
</>
);
}
عدم تغير الحالة عن إجراء تحديث عليها
إن أعيدت القيمة نفسها من خطافٍ مخفِّضٍ (Reducer Hook) والتي تمثِّل الحالة الحالية، فلن تتكبد React عناء تصيير الابن أو تنفيذ التأثيرات. (تستعمل React الخوارزمية Object.is لإجراء عملية الموازنة.)
لاحظ أن React قد تظل بحاجة إلى عرض المكون المحدد مرة أخرى قبل أن تنصرف. لا ينبغي أن يكون هذا مصدر قلق لأن React لن تتعمق دون داع في الشجرة. وإذا كنت تجري عمليات حسابية مكلفة أثناء التصيير، فيمكنك تحسينها باستخدام useMemo
.
useCallback
const memoizedCallback = useCallback(
() => {
doSomething(a, b);
},
[a, b],
);
يعيد هذا الخطاف قيمةً مُستظهَرةً (memoized value).
يُمرَّر إلى هذا الخطاف رد نداء سطري (inline callback) ومصفوفة من المدخلات، ويعيد نسخة مُستظهَرة (memoized version، أي محفوظة دون إعادة تكرار العملية) من رد النداء المعطى والذي يتغير إن تغيرت قيمة إحدى مدخلاته فقط. هذا السلوك مفيدٌ للغاية عند تمرير ردود نداء لتحسين المكونات الأبناء الي تعتمد على المساواة المرجعية (reference equality) لمنع عمليات التصيير الغير ضرورية (مثل shouldComponentUpdate).
الاستدعاء useCallback(fn, inputs)
مكافئ للاستدعاء useMemo(() => fn, inputs)
.
ملاحظة: مصفوفة المدخلات لا تُمرَّر كوسائط إلى رد النداء. نظريًّا، إليك ما الذي تمثله: كل قيمة أشير إليها داخل رد النداء يجب أن تظهر أيضًا في مصفوفة المدخلات. في المستقبل، قد يصبح المصرِّف متقدمًا بما فيه الكفاية لإنشاء هذه المصفوفة تلقائيًا.
نوصي باستخدام قاعدة التبعيات الشاملة كجزء من الحزمة eslint-plugin-reaction-hooks. إذ تطلق تحذيرا في حال حدّدت التبعيات بصورة غير صحيحة مع اقتراح الحل.
useMemo
const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);
يعيد هذا الخطاف قيمةً مُستظهَرةً (memoized value).
يُمرَّر إلى الخطاف useMemo
دالة إنشاء (create function) ومصفوفة من المدخلات. سيحسب الخطاف القيمة المُستظهَرة عند تغيِّر إحدى المدخلات فقط. يساعد هذا التحسين في تجنب إعادة إجراء عمليات الحساب المستهلكة للأداء عند كل تصيير.
تذكر أنَّ الدالة المُمرَّرة إلى الخطاف useMemo
تُنفَّذ أثناء عملية التصيير. لا تفعل أي شيء إضافي لا تفعله عادةً أثناء عملية التصيير. على سبيل المثال، التأثيرات الجانبية تنتمي إلى الخطاف useEffect
وليس إلى الخطاف useMemo
.
إن لم تُعطَ أية مصفوفة، ستُحسَب قيمةٌ جديدةٌ متى ما مُرِّرت نسخة دالة جديدة كأول وسيط. (أو في كل عملية تصيير مع دالة سطرية [inline function])
يمكنك الاعتماد على الخطاف useMemo
لتحسين الأداء، وليس لضمان الدلالات (semantic guarantee). في المستقبل، قد تختار React بأن "تنسى" بعض القيم المُستظهَرة (المحفوظة) وتعيد حسابها من جديد في عملية التصيير التالية وذلك لتحرير الذاكرة لمكونات غير ظاهرة على الشاشة مثلًا. اكتب أولًا شيفرتك لتعمل بشكل صحيح دون الخطاف useMemo
، ومن ثمَّ أضفه لتحسين الأداء.
ملاحظة: مصفوفة المدخلات لا تُمرَّر كوسائط إلى الدالة. نظريًّا، إليك ما الذي تمثله: كل قيمة أُشيِر إليها داخل الدالة يجب أن تظهر أيضًا في مصفوفة المدخلات. في المستقبل، قد يصبح المصرِّف متقدمًا بما فيه الكفاية لإنشاء هذه المصفوفة تلقائيًا.
نوصي باستخدام قاعدة التبعيات الشاملة كجزء من الحزمة eslint-plugin-reaction-hooks. إذ تطلق تحذيرا في حال حدّدت التبعيات بصورة غير صحيحة مع اقتراح الحل.
useRef
const refContainer = useRef(initialValue);
يعيد هذا الخطاف كائنًا مرجعيًّا قابلًا للتعديل (mutable ref object) تُهيَّأ الخاصية .current
فيه إلى قيمة الوسيط المُمرَّر (أي الوسيط initialValue
). قد يبقى الكائن المعاد حتى كامل دورة حياة المكون.
حالة الاستعمال الشائعة لهذا الخطاف هي الحاجة إلى الوصول إلى ابنٍ بشكل إلزامي:
function TextInputWithFocusButton() {
const inputEl = useRef(null);
const onButtonClick = () => {
// (mounted) إلى عنصر الإدخال النصي الموصول `current` يشير
inputEl.current.focus();
};
return (
<>
<input ref={inputEl} type="text" />
<button onClick={onButtonClick}>Focus the input</button>
</>
);
}
تعد useRef
أساسا "صندوقًا" يمكنه تخزين قيمة قابلة للتغيير في خاصيته .current
.
قد تكون معتادًا على استخدام المراجع كوسيلة للوصول إلى شجرة DOM. إذا مرّرت كائنا مرجعيا إلى React مع القيمة
.current
إلى عقدة DOM المقابلة كلما تغيرت تلك العقدة.
انتبه إلى أنَّ نفع الخطاف useRef
يتجاوز الخاصية ref
، إذ هو مفيدٌ جدًا في الإبقاء على أية قيمة قابلة للتعديل في متناول اليد. بشكل مشابه لكيفية استعمال حقول النسخ (instance fields) في الأصناف.
يعمل هذا بنجاح لأنّ useRef()
تنشئ كائن JavaScript كاملًا. الاختلاف الوحيد بين useRef()
وإنشاء كائن {current: ...}
بنفسك هو أن useRef
ستمنحك الكائن المرجعي ref نفسه عند كل تصيير.
ضع في اعتبارك أن useRef
لا تخطرك عندما يتغير محتواها. إذ أنّ تعديل الخاصية .current
لا يؤدي إلى إعادة التصيير. إذا كنت تريد تنفيذ شيفرة برمجية عندما تربط React مرجعا أو تفصله عن عقدة DOM، فقد ترغب في استخدام رد نداء مرجعي ( callback ref) بدلاً من ذلك.
useImperativeHandle
useImperativeHandle(ref, createHandle, [inputs])
ref
. كما هو الحال دومًا، الشيفرة الأمرية التي تستعمل المراجع (refs) يجب أن تُتجنَّب في أغلب الحالات.
الخطاف useImperativeHandle
يجب أن يُستعمَل مع forwardRef
بالشكل التالي:function FancyInput(props, ref) {
const inputRef = useRef();
useImperativeHandle(ref, () => ({
focus: () => {
inputRef.current.focus();
}
}));
return <input ref={inputRef} ... />;
}
FancyInput = forwardRef(FancyInput);
<FancyInput ref={fancyInputRef} />
قادرًا على استدعاء fancyInputRef.current.focus()
.
useLayoutEffect
توقيع هذا الخطاف مماثل تمامًا للخطاف useEffect
ولكن يُنفَّذ بشكل متزامن بعد إجراء كل التعديلات على DOM. استعمل هذا الخطاف لقراءة التخطيط (layout) من شجرة DOM وإعادة التصيير بشكل متزامن. ستُجرَى أية تحديثات مجدولة داخل الخطاف useLayoutEffect
بشكل متزامن قبل أن يملك المتصفح فرصةً لإجراء عملية الرسم.
يفضل استعمال الخطاف useEffect
الأساسي متى ما أمكنك ذلك لتجنب حجز التحديثات البصرية.
نصيحة: إن كنت تستبدل شيفرة كتُبَت عبر مكون صنف أو هجرت مكون صنف وأردت استعمال الخطافات، يُنفَّذ الخطاف useLayoutEffect
في نفس المرحلة التي ينفَّذ فيها التابعان componentDidMount
و componentDidUpdate
، نوصي بالبدء باستخدام useEffect
أولاً، وإن حدثت أي مشكلة فجرب استخدام useLayoutEffect
إذا كنت تستخدم تصييرا بالخادم، فتذكر أنه لا يمكن تنفيذ useLayoutEffect
ولا useEffect
حتى يتم تنزيل JavaScript. لهذا تطلق React تحذيرا إن احتوى مكوّن مُصيّر من الخادم على useLayoutEffect
. لإصلاح ذلك، فإما أن تنقل المنطق البرمجي إلى useEffect
(إذا لم يكن ذلك ضروريًا للتصيير الأول)، أو أخّر عرض ذلك المكون حتى يُصيّر العميل (إذا ظهرت بنية HTML مكسورة قبل تنفيذ useLayoutEffect
).
لاستبعاد مكوّن يحتاج إلى تأثيرات تخطيطية من شيفرة HTML المُصيّرة من الخادم، يمكنك تصييرها شرطيا باستخدام showChild && <Child />
وإرجاء إظهارها باستخدام useEffect(() => { setShowChild(true); }, [])
. بهذه الطريقة، لن تبدو واجهة المستخدم مكسورة قبل ملء البيانات (hydration).
useDebugValue
useDebugValue(value)
useFriendStatus
المخصص الذي شرحناه في صفحة "بناء خطاف خاص بك":function useFriendStatus(friendID) {
const [isOnline, setIsOnline] = useState(null);
// ...
// بجانب هذا الخطاف (DevTools) اظهار تسمية في أدوات التطوير
// "FriendStatus: Online" أي
useDebugValue(isOnline ? 'Online' : 'Offline');
return isOnline;
}
الصيغة المؤجلة لقيم التنقيح
في بعض الحالات، قد تستهلك عملية تنسيق قيمة لإظهارها الكثير من الأداء. أضف إلى ذلك أنَّها غير ضرورية إلا إذا كان يجري فحص خطاف محدَّد.
لذلك السبب، يقبل الخطاف useDebugValue
دالة تنسيق يمكن تمريرها كوسيطٍ ثانٍ اختياريًّا. تُستدعَى هذا الدالة فقط إن كان الخطاف قيد الفحص (inspect)، ويمرَّر إليها قيمة التنقيح كوسيط ويجب أن تعيد قيمة منسقة قابلة للعرض.
Date
أن يتجنب استدعاء الدالة toDateString
بشكل غير ضروري عبر تمرير المنسِّق التالي:useDebugValue(date, date => date.toDateString());
انظر أيضًا
- مدخل إلى الخطافات
- لمحة خاطفة عن الخطافات
- استعمال خطاف الحالة
- استعمال خطاف التأثير
- قواعد استعمال الخطافات
- بناء خطاف خاص بك
- الأسئلة الشائعة حول الخطافات