{"id":17961446,"url":"https://github.com/rafaballerini/reacthooks","last_synced_at":"2025-04-06T11:10:32.791Z","repository":{"id":41261744,"uuid":"354951333","full_name":"rafaballerini/ReactHooks","owner":"rafaballerini","description":"Um guia para entender os diferentes Hooks em React!","archived":false,"fork":false,"pushed_at":"2022-10-12T04:36:56.000Z","size":273,"stargazers_count":357,"open_issues_count":3,"forks_count":73,"subscribers_count":5,"default_branch":"master","last_synced_at":"2024-07-31T19:39:40.046Z","etag":null,"topics":["react","react-component","react-hooks","react-tutorial","reactjs"],"latest_commit_sha":null,"homepage":"","language":"JavaScript","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":null,"status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/rafaballerini.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":null,"funding":null,"license":null,"code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null}},"created_at":"2021-04-05T19:38:43.000Z","updated_at":"2024-07-25T08:16:57.000Z","dependencies_parsed_at":"2023-01-19T21:45:15.151Z","dependency_job_id":null,"html_url":"https://github.com/rafaballerini/ReactHooks","commit_stats":null,"previous_names":[],"tags_count":0,"template":false,"template_full_name":null,"repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rafaballerini%2FReactHooks","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rafaballerini%2FReactHooks/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rafaballerini%2FReactHooks/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rafaballerini%2FReactHooks/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/rafaballerini","download_url":"https://codeload.github.com/rafaballerini/ReactHooks/tar.gz/refs/heads/master","host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":247471521,"owners_count":20944158,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2022-07-04T15:15:14.044Z","host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["react","react-component","react-hooks","react-tutorial","reactjs"],"created_at":"2024-10-29T11:09:21.403Z","updated_at":"2025-04-06T11:10:32.772Z","avatar_url":"https://github.com/rafaballerini.png","language":"JavaScript","funding_links":[],"categories":[],"sub_categories":[],"readme":"# React Hooks \u003cimg src=\"./public/React.svg.png\" width=\"150px\" style=\"position: absolute; right: 10px\"\u003e\n\nEntendendo como funcionam alguns Hooks do React\n\u003chr\u003e\n\n### Como rodar o projeto:\n\n**Você pode clonar o projeto e rodá-lo localmente seguindo os passos abaixo**\n\n1. `git clone https://github.com/rafaballerini/ReactHooks.git` para clonar o projeto\n2. `yarn` para instalar as dependências do projeto\n3. `yarn start`\n4. Acessar [http://localhost:3000](http://localhost:3000) no navegador\n\n### Como testar cada um dos hooks:\n1. Abra o arquivo `App.js`. É possível perceber que existem várias linhas comentadas, tanto na parte de importações, quando dentro da função.\n2. Descomente a importação referente ao hook que deseja testar e também a linha dentro da função (há outro comentário no final da linha indicando a qual hook ela pertence).\n\n\u003chr\u003e\n\n## Explicação de cada Hook:\n\n### useState\nÉ uma função para controle de estado:\n- **recebe um parâmetro** (valor inicial desse estado) \n- **retorna uma lista com 2 variáveis** (a primeira é o valor do estado em si e a segunda é a função que atualiza esse estado)\n\n```js\nconst [state, setState] = useState(0);\n```\n\nO `setState` será usado para atualizar os valores do estado, por exemplo:\n\n```js\n  function increment(){\n    setState(state + 1)\n  }\n```\n\n### useEffect\n- **recebe dois parâmetros:** uma função e uma lista de dependências. Quando algum elemento dessa lista for alterado, a função é executada automaticamente.\n- **o retorno da função pode ser uma função.** Se for, ela será executada quando o componente for desmontado\n\n```js\n  useEffect(() =\u003e {\n    console.log(state)\n  }, [state])\n```\n\nQuando a lista de dependências estiver vazia, a função será executada no momento que o componente for renderizado.\nÉ executado de forma assíncrona depois de uma renderização na tela.\n\n### useContext\nÉ uma forma de mais de um componente ter acesso a uma funcionalidade/lógica do programa.\nPra isso é criado uma `const Context` usando o `React.createContext`.\n\n```js\nconst Context = createContext()\n```\n\nCriamos um componente que será o provedor dos dados que tem no nosso context.\n\n```js\nexport function PageContextProvider({children}) {\n  const [state, setState] = useState(0);\n\n  const increment = useCallback(()=\u003e{\n    setState(state + 1)\n  },[state])\n\n  const decrement = useCallback(()=\u003e{\n    setState(state - 1)\n  },[state])\n\n  const handleChange = useCallback((event)=\u003e{\n    setState(Number(event.target.value))\n  },[state])\n\n  return (\n    \u003cContext.Provider value={{\n        state, \n        increment, \n        decrement, \n        handleChange}\n        }\u003e\n        {children}\n    \u003c/Context.Provider\u003e\n  )\n}\n```\nNote que esse `Context.Provider` recebe a propriedade `children` e utiliza essa propriedade para representar todos os componentes.\n\n```js\n\u003cPageContextProvider\u003e\n    \u003cPageInputContext/\u003e\n\u003c/PageContextProvider\u003e\n```\n\n`PageInputContext` é um componente filho de `PageContextProvider` e para acessar as propriedades que ele envia é preciso chamar o `useContext` passando como parâmetro o `Context`, criado com o `React.createContext`\n\n```js\nexport default function PageInputContext(){\n    const {state, increment, decrement, handleChange} = useContext(Context)\n}\n```\n\n### useReducer\nNo `useState` a lógica de atualização de dados fica dentro de onde ele foi chamado, já no `UseReducer` a lógica ficará em uma função, como essa `reducer` abaixo.\n\n```js\nfunction reducer(state, action) {\n  switch (action.type) {\n    case \"increment\":\n      return state + 1;\n    case \"decrement\":\n      return state - 1;\n    default:\n      return action.newState;\n  }\n}\n```\n\n- **recebe até 3 parâmetros:** o primeiro é a função que altera o estado, o segundo é o estado inicial e o terceiro é uma função init caso seja necessário executar alguma coisa no momento que o estado é criado\n- **retorna uma lista com 2 elementos:** o valor do estado em si e o dispatch, que é a forma como iremos chamar a função para atualizar o estado, ela que dispara a chamada da função reducer\n\n```js\nconst [state, dispatch] = useReducer(reducer, 0);\n```\n\nNormalmente é melhor utilizar o `useReducer` quando se tem uma lógica complexa no state que envolve múltiplos subvalores ou quando o próximo state depende do anterior. \nEle também permite otimizar performance para alguns componentes, pois você pode passar um `dispatch` ao invés de `callbacks`.\n\n```js\n\u003cbutton onClick={() =\u003e dispatch({ type: \"decrement\" })}\u003e\n```\n\n### useCallback\nPara cada alteração no valor do `state`, criamos um `callback` diferente: `increment`, `decrement` e `handleChange`\n- **recebe 2 parâmetros:** uma função e uma lista de dependências. Essa função é [memorizada](https://en.wikipedia.org/wiki/Memoization) e apenas será rerenderezida quando um dos valores da lista de dependências for alterado.\n\n```js\nconst increment = useCallback(()=\u003e{\n    setState(state + 1)\n},[state])\n\nconst decrement = useCallback(()=\u003e{\n    setState(state - 1)\n},[state])\n\nconst handleChange = useCallback((event)=\u003e{\n    setState(Number(event.target.value))\n},[state])\n```\n\nA função não ser recarregada toda hora facilita muito o processamento.\n\n### useMemo\nAssim como o `useCallback`:\n- **recebe 2 parâmetros:** uma função e uma lista de dependências. \n\n```js\nconst memorizedValue = useMemo(() =\u003e {\n    if(state \u003e state2){\n      return 'Maior'\n    }else if(state \u003c state2){\n      return 'Menor'\n    }else{\n      return 'Igual'\n    }\n}, [state, state2])\n```\n\nO valor de retorno da função é [memorizado](https://en.wikipedia.org/wiki/Memoization) e apenas será recarregado quando um dos valores da lista de dependências forem alterados.\n\n### useRef\n- **retorna a referência de um objeto mutável** e que existirá durante toda a vida do componente. \n\n```js\nexport default function InputTexto() {\n    const inputRef = useRef(null);\n    const focaInput = () =\u003e {\n      inputRef.current.focus();\n    };\n    return (\n      \u003cdiv className=\"content refContent\"\u003e\n        \u003cinput ref={inputRef} type=\"text\" /\u003e\n        \u003cbutton className=\"focusButton\" onClick={focaInput}\u003eDá foco no input\u003c/button\u003e\n      \u003c/div\u003e\n    );\n}\n```\n\nPara acessar o objeto é necessário usar a propriedade `.current` do `useRef`.\n\n### useImperativeHandle\nUtilizado para passar um componente `ref` para um componente pai, devendo ser combinado com o `fowardRef`.\n\n```js\nfunction InputTextoImperative(props, ref) {\n    \n    const inputRef = useRef(null);\n\n    useImperativeHandle(ref, () =\u003e ({\n        focus: () =\u003e {\n          inputRef.current.focus();\n        }, \n        console: () =\u003e {\n          console.log('teste')\n        }\n    }));\n\n    return (\n      \u003cdiv className=\"content\"\u003e\n        \u003cinput ref={inputRef} type=\"text\" /\u003e\n      \u003c/div\u003e\n    );\n}\n\nexport default forwardRef(InputTextoImperative);\n```\n\nVocê pode colocar um botão em `App.js` (componente pai) e ele chamar a função do componente filho.\n\n```js\n\u003cInputTextoImperative ref={inputRef}/\u003e \n\u003cbutton className=\"focusButton\" onClick={focaInput}\u003eDá foco no input\u003c/button\u003e\n```\n\n### useLayoutEffect\nÉ muito parecido com o `useEffect`, porém é disparado de forma **síncrona** após todas as mudanças no DOM e antes de aparecer qualquer coisa na tela. \n\n```js\nuseLayoutEffect(() =\u003e {\n    console.log('layout');\n}, [])\n```\n\nÉ preferível utilizar o `useEffect` para evitar bloqueio de atualizações visuais.\n\n**Quando usar?** \nVocê vai perceber que o elemento ficará piscando/executando várias vezes quando utilizar o `useEffect`, nesse caso o ideal é alterar para `useLayoutEffect`.\n\n### useDebugValue\nUtilizado em conjunto com a extensão [React Dev Tools](https://chrome.google.com/webstore/detail/react-developer-tools/fmkadmapgofadopljbjfkapdkoienihi) para mostrar o conteúdo de algum estado, como se fosse um `console.log`.\n\n```js\nfunction useAnalyzeState(state) {\n  useDebugValue(`Valor do state = ${state}`);\n  return state;\n}\n```\n\nComo o próprio nome diz, é utilizado para fazer debug.\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frafaballerini%2Freacthooks","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Frafaballerini%2Freacthooks","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frafaballerini%2Freacthooks/lists"}