DEV Community

Cover image for Api de Contexto em REACT é muito doido!
Jeferson 'Shin' Leite Borges
Jeferson 'Shin' Leite Borges

Posted on • Edited on

Api de Contexto em REACT é muito doido!

This is in portuguese, wanna know why? Click here!

Deixa eu explicar primeiro para dar um contexto no que vamos tratar... Pegou? Contexto? Essa foi a única, prometo.

Mas então...

O que é contexto? E porque eu tenho que me importar com ele? E para onde ele vai?

Beleza, vamos imaginar que você tem a seguinte estrutura:

/* App.js */ const App = () => { // Faz algo muito loco aqui e cria uma array para os menus return ( <div> <Menu lista={arrayDeMenus} /> {/*resto da sua aplicação*/} </div> ); }; 
Enter fullscreen mode Exit fullscreen mode
/* Menu.js */ const Menu = (props) => ( <ul> {props.arrayDeMenus.map(menu => ( <MenuItem icon={menu.icon}>{menu.nome}</MenuItem> ))} </ul> ) 
Enter fullscreen mode Exit fullscreen mode
/* MenuItem.js */ const MenuItem = (props) => ( <li> <i>{props.icon} </i> <p>{props.children}</p> </li> ) 
Enter fullscreen mode Exit fullscreen mode

Beleza, sacou o código né? Sabe o nome disso? Props Hell, ou traduzindo daquele jeito, Inferno de Propriedades, e aí? Como resolve isso? Vamos ficar parado e deixar para o próximo resolver isso?

Claro que não, já temos uma solução para isso, e ela se chama contexto, dessa forma a aplicação inteira pode se beneficiar dessa estrutura e somente quem precisa de algo acessa somente o que precisa.

Mas toma cuidado lindo, porque tu sabe né? Só coloca no contexto o que precisa, porque o contexto com 10mb de informação não ajuda o dispositivo do guri ali que tem um celular low end, então só usa o que precisa, deixa o mais liso possível.

Então vamos resolver o problema, mas agora usando o contexto?
Beleza então!

/* index.js */ export const ContextAPI = createContext(); const menu = [ {nome: "Perfil", icon: "😀"}, {nome: "Configurações", icon: "💻"}, {nome: "Sair", icon: "🔴"}, ] reactDom.render( <ContextAPI.Provider value={menu}> <App /> </ContextAPI.Provider>, document.getElementById("root") ); 
Enter fullscreen mode Exit fullscreen mode
/* App.js */ const App = () => { // Tua aplicação faz o que precisa e esquece do menu, porque ele já existe no index.js! return ( <div> <Menu /> {/*resto da sua aplicação*/} </div> ); }; 
Enter fullscreen mode Exit fullscreen mode
/* Menu.js */ const Menu = (props) => { const contexto = useContext(ContextAPI) return ( <ul> {contexto.map(menu => ( <MenuItem icon={menu.icon}>{menu.nome}</MenuItem> ))} </ul> ) } 
Enter fullscreen mode Exit fullscreen mode
/* MenuItem.js */ const MenuItem = (props) => ( <li> <i>{props.icon} </i> <p>{props.children}</p> </li> ) 
Enter fullscreen mode Exit fullscreen mode

Como funciona, primeiro de tudo se cria um contexto, ali tá no index.js, tem um contexto criado, e veja só, esse contexto está lindão... Mas ele não tem NADA, isso mesmo NADA.

Mas o contexto então vai dar coisas para o resto da aplicação, ao renderizar o <App/> passamos o provedor e esse lindo aí do provider que irá ter um value, e nesse value é que colocamos o que o contexto vai deixar disponível.

No menu usamos um hook ali bonitão, e esse useContext vai receber um contexto, que está no index e vai colocar como a referência de qual contexto receber a informação. Como o contexto tem uma array, já pode sair usando ela.

Então, viu? O App passa completamente despercebido pelo contexto, então, basicamente a informação pulou do index para o Menu, isso é lindo? Eu sei, eu sei. Mas calma, isso é só o começo.

Beleza, quer algo mais legal? Bora fazer um hook de contexto custom? Bora fazer esse contexto ficar ainda mais dinâmico e brincar com um wanna be redux no meio caminho?

Saca esse aqui então:

/* index.js */ reactDom.render( <CustomContext> <App /> </CustomContext>, document.getElementById("root") ); 
Enter fullscreen mode Exit fullscreen mode
/* context.js */ const InitialState = { menu: [ { nome: "Perfil", icon: "😀" }, { nome: "Configurações", icon: "💻" }, { nome: "Sair", icon: "🔴" }, ], }; const AppContext = createContext(InitialState); const CustomContext = ({ children }) => { const [state, dispatch] = useReducer(reducer, InitialState); return ( <AppContext.Provider value={{ state, dispatch }}> {children} </AppContext.Provider> ); }; 
Enter fullscreen mode Exit fullscreen mode
/* reducer.js */ const reducer = (state, { type, payload }) => { switch (type) { case "MENU": return { ...state, menu: [...state.menu, payload], }; default: return state; } }; 
Enter fullscreen mode Exit fullscreen mode
/* useActions.js */ const useActions = () => { const { state, dispatch } = useContext(AppContext); const anotherMenu = async (menu) => { dispatch({ type: "MENU", payload: {menu, icon: "🤯"}}); return; }; return { state, anotherMenu, }; }; 
Enter fullscreen mode Exit fullscreen mode
/* App.js */ const App = () => { const { anotherMenu } = useActions(); // Se tua cabeça não explodir eu não sei o que vai fazer! return ( <div> <Menu /> <button onClick={() => anotherMenu("Cooontexto")} > Novo Menu </button> {/*resto da sua aplicação*/} </div> ); }; 
Enter fullscreen mode Exit fullscreen mode
/* Menu.js */ const Menu = (props) => { const { state } = useActions(); return ( <ul> {state.menu.map((menu) => ( <MenuItem icon={menu.icon}>{menu.nome}</MenuItem> ))} </ul> ); }; 
Enter fullscreen mode Exit fullscreen mode
/* MenuItem.js */ const MenuItem = (props) => ( <li> <i>{props.icon} </i> <p>{props.children}</p> </li> ); 
Enter fullscreen mode Exit fullscreen mode

Tá beleza, duplica essa aba aqui e coloca o código lado a lado que a pancada na mente é forte! Bora então e vamos com cuidado e por partes, beleza?

Primeira coisa, temos o contexto, ele vai ser apenas um preparação de campo, ele vai dar o início nesse trem aqui. Ele é responsável por dar o estado inicial da aplicação, então coloca ali tudo o que não precisa carregar de forma externa.

Ele também vai envelopar o index da aplicação para poder passar o contexto.

Agora vem a segunda parte, o reducer, esse aqui é perigoso, mas tu precisa entender o que ele faz direito, senão dá ruim. Certo então, vamos lá entender o que isso aqui faz.

Mimimimi, tem um switch case aqui!

Tem sim e vai ficar, eu também reclamei, tu vai reclamar, e vai engolir essa calado. Estamos entendidos? Beleza, mais pra frente tu vai entender porque precisa do switch aqui. Mas ele é para saber qual alteração de estado deve ser feita.

No momento tem apenas o "MENU", mas pode (e provável que vá) ter várias, algumas dezenas de alterações de estado.

Mas o que ele altera? Ele vai mudar a informação de maneira síncrona com o estado da aplicação. Então NADA DE FAZER FETCH AQUI! Pensou no async await, também não rola isso é só açúcar sintático para operações assíncronas. Ficou claro? Certo, se precisar usar o reducer para limar informação, alterar ela, converter de string para number, faz tudo aqui. Ele é responsável por atualizar o estado da aplicação.

Repara que ele sempre TEM que retornar o estado, beleza, se retornar null toda a aplicação quebra. Então olha o que faz no reducer!

Ok, vamos a parte legal, o nosso hook. Reparou no nome? Tem o use na frente não tem? Baaaah tchê garoto, primeiro hook custom que coloca para frente, chega a dar um orgulho que só!

Então, o que o useActions faz? Ele vai conceder ações para a aplicação. Ou seja, se quer alterar o contexto da aplicação usa uma ação para alterar esse estado. Essa função do useActions vai retornar várias funções para o usuário brincar, ele também retorna o estado, vai que precisa receber o estado não é mesmo?

Então é aqui que o mundo assíncrono acontece, aqui pode usar FETCH, pode usar await, pode fazer promise, faz o câmbal aqui, pode dar o louco forte e sair girando. Mas entenda uma coisa: usa o dispatch para atualizar o estado da aplicação.

Então já entendeu né. Fez o fetch, recebeu informação do backend, larga um dispatch para atualizar o estado. Mas repara, o dispatch precisa receber sempre um object com duas coisas (pode ter mais, só que aí tu se complica fazendo isso). Quais coisas?

type e payload, então já sabe, usa o type para passar para o que vai bater no switch, e quando o reducer pegar o switch certo ele vai colocar a informação do payload dentro do estado. Belezura, mas como vamos usar?

Olha que lindo, no App e Menu já usamos isso. Manja essa, no App rodamos o useActions() para receber a função que altera o estado, e no Menu rodamos novamente para receber o contexto da aplicação.

Fala sério, tu nunca pensou que iria fazer um redux em tão pouco né? E manja mais essa, tudo em hooks porque somos tudo fino e elegante nesse Javascript.

Por hoje é isso, tem aí material até não aguentar mais o buxo. Ficou com vontade de copiar tudo isso? Beleza, pega esse snippet aqui e guarda a mansa.

Curtiu né, achou que tava brincando né!

(()=>{})()

Top comments (0)