你应该会喜欢的5个自定义Hook

文本已经过原作者 Grégory D'Angelo 授权翻译。

创新互联建站专业为企业提供猇亭网站建设、猇亭做网站、猇亭网站设计、猇亭网站制作等企业网站建设、网页设计与制作、猇亭企业网站模板建站服务,十年猇亭做网站经验,不只是建网站,更提供有价值的思路和整体网络服务。

React hooks

React hooks 已经在16.8版本引入到库中。它允许我们在函数组件中使用状态和其他React特性,这样我们甚至不需要再编写类组件。

实际上,Hooks 远不止于此。

Hooks 可以将组件内的逻辑组织成可重用的独立单元。

Hooks 非常适合 React 组件模型和构建应用程序的新方法。Hooks 可以覆盖类的所有用例,同时在整个应用程序中提供更多的提取、测试和重用代码的灵活性。

构建自己的自定义React钩子,可以轻松地在应用程序的所有组件甚至不同应用程序之间共享特性,这样我们就不必重复自己的工作,从而提高构建React应用程序的效率。

现在,来看看我在开发中最常用的5个自定义钩子,并头开始重新创建它们,这样你就能够真正理解它们的工作方式,并确切地了解如何使用它们来提高生产率和加快开发过程。

我们直接开始创建我们的第一个自定义React Hooks。

useFetch

获取数据是我每次创建React应用时都会做的事情。我甚至在一个应用程序中进行了好多个这样的重复获取。

不管我们选择哪种方式来获取数据,Axios、Fetch API,还是其他,我们很有可能在React组件序中一次又一次地编写相同的代码。

因此,我们看看如何构建一个简单但有用的自定义 Hook,以便在需要在应用程序内部获取数据时调用该 Hook。

okk,这个 Hook 我们叫它 useFetch。

这个 Hook 接受两个参数,一个是获取数据所需查询的URL,另一个是表示要应用于请求的选项的对象。

 
 
 
 
  1. import { useState, useEffect } from 'react'; 
  2.  
  3. const useFetch = (url = '', options = null) => {}; 
  4.  
  5. export default useFetch; 

获取数据是一个副作用。因此,我们应该使用useEffect Hook 来执行查询。

在本例中,我们使用 Fetch API来发出请求。我们会传递URL和 options。一旦 Promise 被解决,我们就通过解析响应体来检索数据。为此,我们使用json()方法。

然后,我们只需要将它存储在一个React state 变量中。

 
 
 
 
  1. import { useState, useEffect } from 'react'; 
  2.  
  3. const useFetch = (url = '', options = null) => { 
  4.   const [data, setData] = useState(null); 
  5.  
  6.   useEffect(() => { 
  7.     fetch(url, options) 
  8.       .then(res => res.json()) 
  9.       .then(data => setData(data)); 
  10.   }, [url, options]); 
  11. }; 
  12.  
  13. export default useFetch; 

这里,我们还需要处理网络错误,以防我们的请求出错。所以我们要用另一个 state 变量来存储错误。这样我们就能从 Hook 中返回它并能够判断是否发生了错误。

 
 
 
 
  1. import { useState, useEffect } from 'react'; 
  2.  
  3. const useFetch = (url = '', options = null) => { 
  4.   const [data, setData] = useState(null); 
  5.   const [error, setError] = useState(null); 
  6.  
  7.   useEffect(() => { 
  8.     fetch(url, options) 
  9.       .then(res => res.json()) 
  10.       .then(data => { 
  11.         if (isMounted) { 
  12.           setData(data); 
  13.           setError(null); 
  14.         } 
  15.       }) 
  16.       .catch(error => { 
  17.         if (isMounted) { 
  18.           setError(error); 
  19.           setData(null); 
  20.         } 
  21.       }); 
  22.   }, [url, options]); 
  23. }; 
  24.  
  25. export default useFetch; 

useFetch返回一个对象,其中包含从URL中获取的数据,如果发生了任何错误,则返回错误。

 
 
 
 
  1. return { error, data }; 

最后,向用户表明异步请求的状态通常是一个好做法,比如在呈现结果之前显示 loading。

因此,我们添加第三个 state 变量来跟踪请求的状态。在请求之前,将loading设置为true,并在请求之后完成后设置为false。

 
 
 
 
  1. const useFetch = (url = '', options = null) => { 
  2.   const [data, setData] = useState(null); 
  3.   const [error, setError] = useState(null); 
  4.   const [loading, setLoading] = useState(false); 
  5.  
  6.   useEffect(() => { 
  7.     setLoading(true); 
  8.  
  9.     fetch(url, options) 
  10.       .then(res => res.json()) 
  11.       .then(data => { 
  12.         setData(data); 
  13.         setError(null); 
  14.       }) 
  15.       .catch(error => { 
  16.         setError(error); 
  17.         setData(null); 
  18.       }) 
  19.       .finally(() => setLoading(false)); 
  20.   }, [url, options]); 
  21.  
  22.   return { error, data }; 
  23. }; 

现在,我们可以返回 loading 变量,以便在请求运行时在组件中使用它来呈现一个 loading,方便用户知道我们正在获取他们所请求的数据。

 
 
 
 
  1. return { loading, error, data }; 

在使用 userFetch 之前,我们还有一件事。

我们需要检查使用我们 Hook 的组件是否仍然被挂载,以更新我们的状态变量。否则,会有内存泄漏。

 
 
 
 
  1. import { useState, useEffect } from 'react'; 
  2.  
  3. const useFetch = (url = '', options = null) => { 
  4.   const [data, setData] = useState(null); 
  5.   const [error, setError] = useState(null); 
  6.   const [loading, setLoading] = useState(false); 
  7.  
  8.   useEffect(() => { 
  9.     let isMounted = true; 
  10.  
  11.     setLoading(true); 
  12.  
  13.     fetch(url, options) 
  14.       .then(res => res.json()) 
  15.       .then(data => { 
  16.         if (isMounted) { 
  17.           setData(data); 
  18.           setError(null); 
  19.         } 
  20.       }) 
  21.       .catch(error => { 
  22.         if (isMounted) { 
  23.           setError(error); 
  24.           setData(null); 
  25.         } 
  26.       }) 
  27.       .finally(() => isMounted && setLoading(false)); 
  28.  
  29.     return () => (isMounted = false); 
  30.   }, [url, options]); 
  31.  
  32.   return { loading, error, data }; 
  33. }; 
  34.  
  35. export default useFetch; 

接下就是怎么用了?

我们只需要传递我们想要检索的资源的URL。从那里,我们得到一个对象,我们可以使用它来渲染我们的应用程序。

 
 
 
 
  1. import useFetch from './useFetch'; 
  2.  
  3. const App = () => { 
  4.   const { loading, error, data = [] } = useFetch( 
  5.     'https://hn.algolia.com/api/v1/search?query=react' 
  6.   ); 
  7.  
  8.   if (error) return 

    Error!

  9.   if (loading) return 

    Loading...

  10.  
  11.   return ( 
  12.     
     
  13.       
       
    •         {data?.hits?.map(item => ( 
    •            
    •             {item.title} 
    •           
    •  
    •         ))} 
    •       
     
  14.     
 
  •   ); 
  • }; 
  •  useEventListener

    这个 Hook 负责在组件内部设置和清理事件监听器。

    这样,我们就不需要每次添加事件监听器,做重复的工作。

    这个函数有几个参数,eventType 事件类型,listener 监听函数,target 监听对象,options 可选参数。

     
     
     
     
    1. import { useEffect, useRef } from 'react'; 
    2.  
    3. const useEventListener = ( 
    4.   eventType = '', 
    5.   listener = () => null, 
    6.   target = null, 
    7.   options = null 
    8. ) => {}; 
    9.  
    10. export default useEventListener; 

    与前一个 Hook 一样,用 useEffect 来添加一个事件监听器。首先,我们需要确保target 是否支持addEventListener方法。否则,我们什么也不做。

     
     
     
     
    1. import { useEffect, useRef } from 'react'; 
    2.  
    3. const useEventListener = ( 
    4.   eventType = '', 
    5.   listener = () => null, 
    6.   target = null, 
    7.   options = null 
    8. ) => { 
    9.  
    10.   useEffect(() => { 
    11.     if (!target?.addEventListener) return; 
    12.   }, [target]); 
    13. }; 
    14.  
    15. export default useEventListener; 

    然后,我们可以添加实际的事件监听器并在卸载函数中删除它。

     
     
     
     
    1. import { useEffect, useRef } from 'react'; 
    2.  
    3. const useEventListener = ( 
    4.   eventType = '', 
    5.   listener = () => null, 
    6.   target = null, 
    7.   options = null 
    8. ) => { 
    9.   useEffect(() => { 
    10.     if (!target?.addEventListener) return; 
    11.  
    12.     target.addEventListener(eventType, listener, options); 
    13.  
    14.     return () => { 
    15.       target.removeEventListener(eventType, listener, options); 
    16.     }; 
    17.   }, [eventType, target, options, listener]); 
    18. }; 
    19.  
    20. export default useEventListener; 

    实际上,我们也会使用一个引用对象来存储和持久化监听器函数。只有当监听器函数发生变化并在事件监听器方法中使用该引用时,我们才会更新该引用。

     
     
     
     
    1. import { useEffect, useRef } from 'react'; 
    2.  
    3. const useEventListener = ( 
    4.   eventType = '', 
    5.   listener = () => null, 
    6.   target = null, 
    7.   options = null 
    8. ) => { 
    9.   const savedListener = useRef(); 
    10.  
    11.   useEffect(() => { 
    12.     savedListener.current = listener; 
    13.   }, [listener]); 
    14.  
    15.   useEffect(() => { 
    16.     if (!target?.addEventListener) return; 
    17.  
    18.     const eventListener = event => savedListener.current(event); 
    19.  
    20.     target.addEventListener(eventType, eventListener, options); 
    21.  
    22.     return () => { 
    23.       target.removeEventListener(eventType, eventListener, options); 
    24.     }; 
    25.   }, [eventType, target, options]); 
    26. }; 
    27.  
    28. export default useEventListener; 

    我们不需要从此 Hook 返回任何内容,因为我们只是侦听事件并运行处理程序函数传入作为参数。

    现在,很容易将事件侦听器添加到我们的组件(例如以下组件)中,以检测DOM元素外部的点击。如果用户单击对话框组件,则在此处关闭对话框组件。

     
     
     
     
    1. import { useRef } from 'react'; 
    2. import ReactDOM from 'react-dom'; 
    3. import { useEventListener } from './hooks'; 
    4.  
    5. const Dialog = ({ show = false, onClose = () => null }) => { 
    6.   const dialogRef = useRef(); 
    7.  
    8.   // Event listener to close dialog on click outside element 
    9.   useEventListener( 
    10.     'mousedown', 
    11.     event => { 
    12.       if (event.defaultPrevented) { 
    13.         return; // Do nothing if the event was already processed 
    14.       } 
    15.       if (dialogRef.current && !dialogRef.current.contains(event.target)) { 
    16.         console.log('Click outside detected -> closing dialog...'); 
    17.         onClose(); 
    18.       } 
    19.     }, 
    20.     window 
    21.   ); 
    22.  
    23.   return show 
    24.     ? ReactDOM.createPortal( 
    25.          
    26.           
    27.             className="relative bg-white rounded-md shadow-card max-h-full max-w-screen-sm w-full animate-zoom-in px-6 py-20" 
    28.             ref={dialogRef} 
    29.           > 
    30.              
    31.               What's up{' '} 
    32.                
    33.                 YouTube 
    34.                
    35.               ? 
    36.             

       
    37.           
     
  •         
  •         document.body 
  •       ) 
  •     : null; 
  • }; 
  •  
  • export default Dialog; 
  •  useLocalStorage

    这个 Hook 主要有两个参数,一个是 key,一个是 value。

     
     
     
     
    1. import { useState } from 'react'; 
    2.  
    3. const useLocalStorage = (key = '', initialValue = '') => {}; 
    4.  
    5. export default useLocalStorage;  

    然后,返回一个数组,类似于使用 useState 获得的数组。因此,此数组将包含有状态值和在将其持久存储在localStorage 中时对其进行更新的函数。

    首先,我们创建将与 localStorage 同步的React状态变量。

     
     
     
     
    1. import { useState } from 'react'; 
    2.  
    3. const useLocalStorage = (key = '', initialValue = '') => { 
    4.   const [state, setState] = useState(() => { 
    5.     try { 
    6.       const item = window.localStorage.getItem(key); 
    7.       return item ? JSON.parse(item) : initialValue; 
    8.     } catch (error) { 
    9.       console.log(error); 
    10.       return initialValue; 
    11.     } 
    12.   }); 
    13. }; 
    14.  
    15. export default useLocalStorage; 

    在这里,我们使用惰性初始化来读取 localStorage 以获取键的值,如果找到该值,则解析该值,否则返回传入的initialValue。

    如果在读取 localStorage 时出现错误,我们只记录一个错误并返回初始值。

    最后,我们需要创建 update 函数来返回它将在localStorage 中存储任何状态的更新,而不是使用useState 返回的默认更新。

     
     
     
     
    1. import { useState } from 'react'; 
    2.  
    3. const useLocalStorage = (key = '', initialValue = '') => { 
    4.   const [state, setState] = useState(() => { 
    5.     try { 
    6.       const item = window.localStorage.getItem(key); 
    7.       return item ? JSON.parse(item) : initialValue; 
    8.     } catch (error) { 
    9.       return initialValue; 
    10.     } 
    11.   }); 
    12.  
    13.   const setLocalStorageState = newState => { 
    14.     try { 
    15.       const newStateValue = 
    16.         typeof newState === 'function' ? newState(state) : newState; 
    17.       setState(newStateValue); 
    18.       window.localStorage.setItem(key, JSON.stringify(newStateValue)); 
    19.     } catch (error) { 
    20.       console.error(`Unable to store new value for ${key} in localStorage.`); 
    21.     } 
    22.   }; 
    23.  
    24.   return [state, setLocalStorageState]; 
    25. }; 
    26.  
    27. export default useLocalStorage; 

    此函数同时更新React状态和 localStorage 中的相应键/值。这里,我们还可以支持函数更新,例如常规的useState hook。

    最后,我们返回状态值和我们的自定义更新函数。

    现在可以使用useLocalStorage hook 将组件中的任何数据持久化到localStorage中。

     
     
     
     
    1. import { useLocalStorage } from './hooks'; 
    2.  
    3. const defaultSettings = { 
    4.   notifications: 'weekly', 
    5. }; 
    6.  
    7. function App() { 
    8.   const [appSettings, setAppSettings] = useLocalStorage( 
    9.     'app-settings', 
    10.     defaultSettings 
    11.   ); 
    12.  
    13.   return ( 
    14.      
    15.        
    16.         Your application's settings:

       
    17.  
    18.         
    19.           value={appSettings.notifications} 
    20.           onChange={e => 
    21.             setAppSettings(settings => ({ 
    22.               ...settings, 
    23.               notifications: e.target.value, 
    24.             })) 
    25.           } 
    26.           className="border border-gray-900 rounded py-2 px-4 " 
    27.         > 
    28.           daily 
    29.           weekly 
    30.           monthly 
    31.          
    32.        
    33.  
    34.       
    35.         onClick={() => setAppSettings(defaultSettings)} 
    36.         className="rounded-md shadow-md py-2 px-6 bg-red-500 text-white uppercase font-medium tracking-wide text-sm leading-8" 
    37.       > 
    38.         Reset settings 
    39.        
    40.      
    41.   ); 
    42.  
    43. export default App; 

     useMediaQuery

    这个 Hook 帮助我们在功能组件中以编程方式测试和监控媒体查询。这是非常有用的,例如,当你需要渲染不同的UI取决于设备的类型或特定的特征。

    我们的 Hook 接受3个参数:

     
     
     
     
    1. import { useState, useCallback, useEffect } from 'react'; 
    2.  
    3. const useMediaQuery = (queries = [], values = [], defaultValue) => {}; 
    4.  
    5. export default useMediaQuery; 

    我们在这个 Hook 中做的第一件事是为每个匹配的媒体查询构建一个媒体查询列表。使用这个数组通过匹配媒体查询来获得相应的值。

     
     
     
     
    1. import { useState, useCallback, useEffect } from 'react'; 
    2.  
    3. const useMediaQuery = (queries = [], values = [], defaultValue) => { 
    4.   const mediaQueryList = queries.map(q => window.matchMedia(q)); 
    5. }; 
    6.  
    7. export default useMediaQuery; 

    为此,我们创建了一个包装在useCallback 中的回调函数。检索列表中第一个匹配的媒体查询的值,如果没有匹配则返回默认值。

     
     
     
     
    1. import { useState, useCallback, useEffect } from 'react'; 
    2.  
    3. const useMediaQuery = (queries = [], values = [], defaultValue) => { 
    4.   const mediaQueryList = queries.map(q => window.matchMedia(q)); 
    5.  
    6.   const getValue = useCallback(() => { 
    7.     const index = mediaQueryList.findIndex(mql => mql.matches); 
    8.     return typeof values[index] !== 'undefined' ? values[index] : defaultValue; 
    9.   }, [mediaQueryList, values, defaultValue]); 
    10. }; 
    11.  
    12. export default useMediaQuery; 

    然后,我们创建一个React状态来存储匹配的值,并使用上面定义的函数来初始化它。

     
     
     
     
    1. import { useState, useCallback, useEffect } from 'react'; 
    2.  
    3. const useMediaQuery = (queries = [], values = [], defaultValue) => { 
    4.   const mediaQueryList = queries.map(q => window.matchMedia(q)); 
    5.  
    6.   const getValue = useCallback(() => { 
    7.     const index = mediaQueryList.findIndex(mql => mql.matches); 
    8.     return typeof values[index] !== 'undefined' ? values[index] : defaultValue; 
    9.   }, [mediaQueryList, values, defaultValue]); 
    10.  
    11.   const [value, setValue] = useState(getValue); 
    12. }; 
    13.  
    14. export default useMediaQuery; 

    最后,我们在 useEffect 中添加一个事件监听器来监听每个媒体查询的更改。当发生变化时,我们运行更新函数。

     
     
     
     
    1. mport { useState, useCallback, useEffect } from 'react'; 
    2.  
    3. const useMediaQuery = (queries = [], values = [], defaultValue) => { 
    4.   const mediaQueryList = queries.map(q => window.matchMedia(q)); 
    5.  
    6.   const getValue = useCallback(() => { 
    7.     const index = mediaQueryList.findIndex(mql => mql.matches); 
    8.     return typeof values[index] !== 'undefined' ? values[index] : defaultValue; 
    9.   }, [mediaQueryList, values, defaultValue]); 
    10.  
    11.   const [value, setValue] = useState(getValue); 
    12.  
    13.   useEffect(() => { 
    14.     const handler = () => setValue(getValue); 
    15.     mediaQueryList.forEach(mql => mql.addEventListener('change', handler)); 
    16.  
    17.     return () => 
    18.       mediaQueryList.forEach(mql => mql.removeEventListener('change', handler)); 
    19.   }, [getValue, mediaQueryList]); 
    20.  
    21.   return value; 
    22. }; 
    23.  
    24. export default useMediaQuery; 

    我最近使用的一个简单的例子是添加一个媒体查询来检查设备是否允许用户悬停在元素上。这样,如果用户可以悬停或应用基本样式,我就可以添加特定的不透明样式。

     
     
     
     
    1. import { useMediaQuery } from './hooks'; 
    2.  
    3. function App() { 
    4.   const canHover = useMediaQuery( 
    5.     // Media queries 
    6.     ['(hover: hover)'], 
    7.     // Values corresponding to the above media queries by array index 
    8.     [true], 
    9.     // Default value 
    10.     false 
    11.   ); 
    12.  
    13.   const canHoverClass = 'opacity-0 hover:opacity-100 transition-opacity'; 
    14.   const defaultClass = 'opacity-100'; 
    15.  
    16.   return ( 
    17.     Hover me! 
    18.   ); 
    19.  
    20. export default App; 

     useDarkMode

    这个是我的最爱。它能轻松快速地将暗模式功能应用于任何React应用程序。

    这个 Hook 主要按需启用和禁用暗模式,将当前状态存储在localStorage 中。

    为此,我们将使用我们刚刚构建的两个钩子:useMediaQuery和useLocalStorage。

    然后,使用“ useLocalStorage”,我们可以在localStorage中初始化,存储和保留当前状态(暗或亮模式)。

     
     
     
     
    1. import { useEffect } from 'react'; 
    2. import useMediaQuery from './useMediaQuery'; 
    3. import useLocalStorage from './useLocalStorage'; 
    4.  
    5. const useDarkMode = () => { 
    6.   const preferDarkMode = useMediaQuery( 
    7.     ['(prefers-color-scheme: dark)'], 
    8.     [true], 
    9.     false 
    10.   ); 
    11. }; 
    12.  
    13. export default useDarkMode; 

    最后一部分是触发副作用,以向document.body元素添加或删除dark类。这样,我们可以简单地将dark样式应用于我们的应用程序。

     
     
     
     
    1. import { useEffect } from 'react'; 
    2. import useMediaQuery from './useMediaQuery'; 
    3. import useLocalStorage from './useLocalStorage'; 
    4.  
    5. const useDarkMode = () => { 
    6.   const preferDarkMode = useMediaQuery( 
    7.     ['(prefers-color-scheme: dark)'], 
    8.     [true], 
    9.     false 
    10.   ); 
    11.  
    12.   const [enabled, setEnabled] = useLocalStorage('dark-mode', preferDarkMode); 
    13.  
    14.   useEffect(() => { 
    15.     if (enabled) { 
    16.       document.body.classList.add('dark'); 
    17.     } else { 
    18.       document.body.classList.remove('dark'); 
    19.     } 
    20.   }, [enabled]); 
    21.  
    22.   return [enabled, setEnabled]; 
    23. }; 
    24.  
    25. export default useDarkMode; 

     ~完,我是小智,我要去刷碗了。

    作者:Grégory D'Angelo 译者:前端小智 来源: dev原文:https://dev.to/alterclass/5-react-custom-hooks-you-should-start-using-explained-5d18

    网站栏目:你应该会喜欢的5个自定义Hook
    本文链接:http://www.csdahua.cn/qtweb/news43/102293.html

    网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

    广告

    声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网

    成都快上网为您推荐相关内容

    网站设计公司知识

    同城分类信息