Global state yönetimi için kullanabileceğimiz bir araçtır. Toolkit ile işleri çok daha kolay bir şekilde yönetebiliriz.
İLk olarak paketimiz şöyle kuralım:
npm install @reduxjs/toolkit react-redux
İlk olarak bir store dosyası oluşturmamız gerekiyor. Bunun için store/
klasöründe örnek bir store dosyası oluşturalım. Bu dosya global değerlerimizi tutacak. Örneğin site teması ve dili için bir store dosyası tanımlayabiliriz.
store/site.js
adında bir dosya oluşturalım ve şu kodları yazalım:
import { createSlice } from '@reduxjs/toolkit'
const initialState = {
theme: 'light',
language: 'tr'
}
export const site = createSlice({
name: 'site',
initialState,
reducers: {
setTheme: (state, action) => {
state.theme = action.payload
},
setLanguage: (state, action) => {
state.language = action.payload
}
},
})
export const { setTheme, setLanguage } = site.actions
export default site.reducer
Ve store/index.js
dosyası oluşturup ilgili store'larımızı şöyle dahil edelim:
import { configureStore } from '@reduxjs/toolkit'
import site from "./site"
const store = configureStore({
reducer: {
site
},
})
export default store
Redux değerlerini provide edebilmek için yine en üst component'de ya da index.js
de şu işlemleri yapalım:
import React from 'react'
import ReactDOM from 'react-dom'
import './index.css'
import App from './App'
import store from './store'
import { Provider } from 'react-redux'
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
document.getElementById('root')
)
Artık component'lerde bu global değerlere erişip kullanmaya çalışalım:
import { useSelector, useDispatch } from "react-redux"
import { setTheme, setLanguage } from "./store/site"
export default function Header() {
const { theme, language } = useSelector(state => state.site)
const dispatch = useDispatch()
const switchTheme = () => {
dispatch(setTheme(theme === 'light' ? 'dark' : 'light'))
}
const switchLanguage = () => {
dispatch(setLanguage(language === 'tr' ? 'en' : 'tr'))
}
return (
<>
<h1>Header</h1>
Tema = {theme} <br />
<button onClick={switchTheme}>Temayı değiştir</button>
<hr />
Dil = {language} <br />
<button onClick={switchLanguage}>Dili değiştir</button>
</>
)
}
Gördüğünüz gibi yapı Context API + reducer kullanımına çok benziyor. Yine bir dispatch()
metodumuz var ve site.js
içinde export ettiğimiz reducer'ları çalıştırmamızı sağlıyor.
Bazen component dışında da state'lere erişmek ya da state'lerin değerlerini değiştirmek isteriz. Örneğin sorguları yönettiğimiz js dosyamızda state'lerden bir değer almamız icap edebilir. Bu gibi durumlarda store
dosyasını çağırıyoruz.
import store from "./store"
store.getState() // stateleri öner
store.dispatch(..) // bir reducer çalıştırmamızı sağlar
Çoğu zaman istekleri yönetmek istediğimizde zorlanabiliriz. Örneğin post'ları çektiğimiz bir sayfamız olsun, birkaç state tanımlayarak başlarız:
const [loading, setLoading] = useState(false)
const [posts, setPosts] = useState(false)
Daha sonra useEffect()
ile component ilk kez yüklendiğinde gidip isteği atarız, ona göre durumları belirleriz.
useEffect(() => {
setLoading(true)
fetchPosts()
.then(res => setPosts(res))
}, [])
Ancak bu süreç biraz yorucu olabilir, query'leri yani sorgularınızı yönetmek için react query kullanmak projelerde işinizi çok kolaylaştıracaktır.
Projemize react-query paketini kurarak başlayalım:
npm i react-query
İlk olarak index.js
de provider'ı sarmalamak gerekiyor.
import { QueryClient, QueryClientProvider } from "react-query"
const queryClient = new QueryClient()
render(
<QueryClientProvider client={queryClient}>
<App />
</QueryClientProvider>
)
Artık sorgularımızı useQuery()
hook'u ile kolayca yönetebiliriz:
const info = useQuery('todos', fetchTodoList)
Burada
fetchTodoList
bir Promise döndüren metod olmalı. Yani sorgular mutlaka promise dönen bir metod olarak tanımlanmalı.
Burada useQuery
bize bir obje dönüyor.
- isLoading - Yüklenme aşamasında true olarak döner
- isError - Denemeler sonucu bir hata olursa true döner
- isSuccess - İşlem başarılıysa true döner
- isIdle - Sorgu disable ise true döner
- error -
isError
true olduğunda hataları döner - data -
isSuccess
true olduğunda verileri döner - isFetching - Bir veri çekiliyorsa (arkaplanda çekilmesi dahil) true döner
Buna bağlı olarak kodumuzu şöyle düzenleyebilirdik:
import { useQuery } from "react-query"
function TodoApp() {
const { isLoading, isError, error, data } = useQuery('todos', fetchTodoList)
if (isLoading) return 'Yükleniyor..'
if (isError) {
return <pre>{JSON.stringify(error, null, 2)}</pre>
}
return <pre>{JSON.stringify(data, null, 2)}</pre>
}
Her sorgu bir isimle tutulur. Yukarıdaki örneğimizde ismi todos
idi. Ve her sorgu key'ler alabilir, bunu şöyle düşünün sayfa değiştiğinde sorgunun yeniden tetiklenmesini istersiniz değil mi? İşte bu gibi durumlarda sorgu key'lerini kullanacaksınız. Bu bir string olabilir, obje olabilir, array olabilir, canınız ne isterse ekleyin :)
useQuery('todos') // ['todos']
useQuery(['todo', page]) // ['todos', 2]
useQuery(['todo', {
page: 1
}]) // ['todos', {page: 1}]
Not: Eğer sorgularınızda bu state'lere bağlı olarak değişecekse method'da bunları belirtmeyi unutmayın.
useQuery()
metoduna 2. parametre olarak sorgu metodumuzu ekliyoruz. Ve dilersek bu sorgu metodumuz içerisinde sorgu key'lerine erişebiliriz.
function fetchTodoList({ queryKey }) {
console.log(queryKey)
}
...
useQuery(['todos', { page: 2 }], fetchTodoList)
Bazen sorgumuzu yazmak ama çalıştırma eylemi için bir aksiyon olarak bunu gerçekleştirmek isteyebiliriz. Örneğin sorgumzu hazırlarız, ancak bir buton'a basıldığında çalışmasını isteyebiliriz. İşte bu gibi durumlar için bekleme modunda kalması adına enabled
değerini kullanabiliriz.
useQuery('todos', getTodos, {
enabled: false
})
Buradaki false
değeri bir state'e bağlı olursa ve butona basınca bu state true olarak güncellenirse sorgumuz çalışacaktır.
Ayrıca useQuery
altında gelen refetch
ile de yeniden sorgulama işlemini yapabilirsiniz.
Bazen kullanıcı tarayıcıda başka bir taba geçtiğinde ya da focus'unu kaybettiğinde ve siteye geri döndüğünde mevcut sorguların yeniden çalıştıırlıp güncel veriyi almasını isteyebilirsiniz. React Query zaten böyle çalışıyor :D Ama bazen de bunu istemeyebilirsiniz, işte refetchOnWindowFocus
ile bunu belirleyebilirsiniz.
useQuery('todos', getTodos, {
refetchOnWindowFocus: false
})
Focus işleminde artık yeniden çekmeye çalışmayacaktır.
Bir istek hata vermeden önce kaç defa yeniden denenmeli bunu bu değer ile belirtebilirsiniz.
useQuery('todos', getTodos, {
retry: 10
})
10 kere denedikten sonra hala hata veriyorsa o zaman hatayı döndürecektir.
Bazen özellikle sayfalama işlemlerinde 2. sayfaya geçerken 1. sayfanın verisi kaybolur. Bu süreci eski veri kaybolmadan yapmak ve yeni veri geldiğinde eskisiyle değiştirilmesini isterseniz bu değeri kullanabilirsiniz.
useQuery('todos', getTodos, {
keepPreviousData: true
})
Sorgularınız artık güncelliğini yitirmişse örneğin yeni bir veri eklediniz ve son halini çektirmek istiyorsunuz, mevcut sorgunuzu invalidate ederek bu işlemi sağlayabilirsiniz.
Bu işlemi daha sonra mutate ile yapmayı öğreneceğiz.
queryClient.invalidateQueries('todos')
queryClient.invalidateQueries(['todos', { page: 1 }])
Örneğin todos
ismiyle istek atıp değerleri çekiyoruz. Peki yeni bir todo eklemek istediğimizde, bu değeri nasıl güncelleyeceğiz?
Ya yukarıda invalidateQueries()
ile isteği geçersiz kılarak yenisini çektireceğiz ki bu saçma olur, ya da şöyle yapacağız:
import { useQueryClient } from "react-query"
const queryClient = useQueryClient()
const submitHandle = async () => {
const todo = await newTodo({
title: 'todo 1',
done: false
})
// mevcut todoları getir
const previousTodos = queryClient.getQueryData('todos')
// Yeni todo ile birlikte güncelle
queryClient.setQueryData('todos', old => [todo, ...old])
}
Sorguların aksine mutation'lar oluşturma/güncelleme/silme işlemlerini yönetmek için kullanılır. Bu amaç içinde, useMutation()
hooku kullanılır. Örnek bir kullanım şöyledir:
import { useMutation } from "react-query"
export default function NewTodo() {
const mutation = useMutation(newTodo)
return (
<>
{mutation.isLoading && 'Ekleniyor..'}
{mutation.isError && 'Bir hata oluştu'}
{mutation.isSuccess && 'Todo başarıyla eklendi'}
<button onClick={() => {
mutation.mutate({
title: 'todo 1',
done: false
})
}}>
Todo Ekle
</button>
</>
)
}
Ayrıca mutate
metoduna 2. parametre olarak bir obje verilip onSuccess
, onError
gibi durumlarda mevcut data güncellemesi yapılabilirdi.
Bunun için bir önceki örneği ele alıp test edebilirsiniz.
Henüz deneysel bir çalışma olsada 2 farklı tabı senkron etmek istediğinizde bu eklentiyi kullanabilirsiniz. Ekstra olarak yüklemenize gerek yok, react-query ile birlikte geliyor. index.js
e şu şekilde çağırıp dahil edebilirsiniz:
import { broadcastQueryClient } from 'react-query/broadcastQueryClient-experimental'
const queryClient = new QueryClient()
broadcastQueryClient({
queryClient,
broadcastChannel: 'my-app',
})
Artık 2 farklı tab react-query ile bir işlem yapıldığında senkron hale gelecek, denemesi bedava :)
React ile proje geliştirirken, dil sistemine de mutlaka ihtiyaç duyacağız. Bu gibi durumlarda bu işin altından hakkıyla kalkmak için react-i18n
paketine ihtiyacımız olacak.
Kurmak için:
npm i react-i18next i18next
İlk olarak i18n.js
dosyası oluşturup şu şekilde kodlarımızı yazalım.
import i18n from "i18next";
import {initReactI18next} from 'react-i18next';
Ve hemen altında ufak bir çeviri ekleyelim ingilizce ve türkçe için.
const resources = {
en: {
translation: {
welcome: 'Welcome to React'
}
},
tr: {
translation: {
welcome: 'React\'e Hoşgeldin'
}
}
}
Ve ayarlarımızı başlatalım.
i18n
.use(initReactI18next)
.init({
resources,
lng: 'en'
})
export default i18n
i18n.js
dosyasını index.js
e çağırdıktan sonra hazırız.
Artık component'ler içinde kullanmaya hazırız. Örneğin App
componentimizde şu şekilde kullanabiliriz:
import { useTranslation } from "react-i18next";
function App() {
const { t, i18n } = useTranslation()
return (
<>
<h1>{t('welcome')}</h1>
<button onClick={() => i18n.changeLanguage(i18n.language === 'en' ? 'tr' : 'en')}>Dili Değiştir</h1>
</>
)
}
Inline olarak dil kodları tanımlamak elbette çok mantıklı değil, public/locales
altında dil dosyalarını ayrı ayrı tutmak isteyebiliriz. Bunun için şu paketi kuracağız:
npm i i18next-http-backend
Ve setup'a şöyle dahil edeceğiz:
import Backend from 'i18next-http-backend';
i18n
.use(Backend)
.use(initReactI18next)
.init({
lng: 'en'
})
Ve örnek dil dosyalarımız tr ve en dilinde olacaksa dosya yapısı şöyle olmalı:
- public/locales/en/translation.json
- public/locales/tr/translation.json
Örnek translation.json
dosyası ise şöyle olmalı:
{
"welcome": "Welcome to React"
}
Eğer react projesi içinde değilde ayrı bir backendden dilleri çekmek istiyorsanız, yukarıdaki adıma ek olarak, init()
metodu içinde şu şekilde bir ayar vermemiz gerekiyor. Diyelim ki backend için dil yükleme metodumuz şu olsun:
https://api.prototurk.com/language/tr
https://api.prototurk.com/language/en
Burada tr
ve en
dinamik olarak şöyle gönderilerek yüklenebilir:
i18n
.use(Backend)
.use(initReactI18next)
.init({
lng: 'en',
backend: {
loadPath: 'https://api.prototurk.com/language/{{lng}}'
}
})
Ek olarak backend tarafında CORS izinleri verilmiş olması gerekiyor. Detaylı bilgi için şuraya bakabilirsiniz.
Varsayılan tarayıcı diline göre dil dosyası yükletmek için şu paketi kurmamız lazım:
npm i i18next-browser-languagedetector
Ve setup'ımız şöyle olmalı:
import LanguageDetector from 'i18next-browser-languagedetector';
i18n
.use(Backend)
.use(LanguageDetector)
.use(initReactI18next)
.init({
fallbackLng: 'en'
})
Artık tarayıcının dil dosyası ne ise onu kullanmaya çalışacak, eğer ilgili dile ait bir dosya yoksa fallbackLng
altındaki mevcut dil dosyasını gösterecektir.
Böyle bir ihtiyaç halinde i18n.js
dosyasını import edip içindeki t
metodunu kullanabilirsiniz:
import i18n from "./i18n"
console.log(i18n.t('welcome'))
withTranslation
higher order function'ı kullanarak component'e prop olarak i18n
ve t
parameterleri geçilebilirdi:
import { withTranslation } from 'react-i18next';
function MyComponent({ t, i18n }) {
return <p>{t('my translated text')}</p>
}
export default withTranslation()(MyComponent);
Dinamik değerlerinizi çeviriye dahil etmek için kullanabilirsiniz. Örneğin dil dosyanızda framework
adını dinamik alacaksanız şu şekilde düzenleyecksiniz:
{
"welcome": "Welcome to {{framework}}"
}
Ve gösterirken de şöyle göndereceksiniz:
t('welcome', { framework: 'React' })
Ayrıca obje olarakda gönderebilirdik:
const framework = {
name: 'React'
}
t('welcome', { framework })
Dil dosyasında şöyle kullanmamız gerekirdi:
{
"welcome": "Welcome to {{framework.name}}"
}