- Mobil Programlama Nedir?
- Mobil Uygulama Geliştirmek için Hangi Seçenekler vardır?
- React Native Nedir?
- React Native Nasıl Çalışır?
- React Native Temelleri
Mobil cihazlar için yazılım uygulamaları oluşturma sürecine mobil programlama diyoruz.
Günümüzde üç çeşit mobil programlama geliştirme seçeneği mevcuttur.
- Hybrid
Tarayıcı motoru üzerinden derlenen uygulama geliştirme yöntemidir.
- Native
İşletim sistemi tarafından direkt derlenebilen uygulama geliştirme yöntemidir.
- Cross Platfrom
Bir Bridge ya da SDK aracılığıyla derlenen uygulama geliştirme yöntemidir.
React Native, hem IOS hem de Android işletim sistemleri için mobil uygulamalar geliştirmemizi sağlayan Javascript tabanlı bir Framework’tür.
React Native ile geliştirmiş bir uygulama Native uygulamaların aksine direk işletim sistemi tarafından değilde işletim sistemi üzerine inşa edilmiş Javascript motoru tarafından derlenir. Bu Javacript motoruna React Native Bridge diyoruz ve 3 ana bölümden oluşuyor.
- UI Thread
Ekranda görüntülenecek bileşenlerin kontrolünü gerçekleştirir.
- JS Thread
Javascript ile geliştirilmiş tüm yapıların kontrolünü gerçekleştirir.
- Native Thread
Telefonun yerel bileşenlerine erişmek istediğimizdeki tüm sistemsel çağrıların kontrolünü gerçekleştirir.
React ve React Native ortamında programlama yaparken kullanılacak sözdiziminin okunmasını veya ifade edilmesini kolaylaştırmak için tasarlanmış formattır.
//Bu şekildeki bir yapıyı...
React.createElement(
MyButton,
{color: 'blue', shadowSize: 2},
'Click Me'
)
//Bu şekile çevirerek daha anlaşılır bir şekilde kullanmamızı sağlar.
<MyButton color="blue" shadowSize={2}>
Click Me
</MyButton>
React ve React Native dünyasınındaki her bir parçadır. Değer alabilen, aldığı değerleri işleyebilen özel yapılardır. Kullanıcı arayüzünü bağımsız, yeniden kullanılabilir parçalara ayırmanıza ve her bir parçayı ayrı ayrı düşünmenize olanak tanır. Kavramsal olarak componentler JavaScript fonksiyonları gibidir. Rastgele girdileri (props) kabul eder ve ekranda neyin görünmesi gerektiğini açıklayan React öğelerini döndürürler.
Fonksiyonel componentler daha basittir. Kendi durumlarını yönetmezler veya React Native tarafından sağlanan yaşam döngüsü yöntemlerine erişimleri yoktur. Tam anlamıyla eski JavaScript fonksiyonlarıdır ve bazen durumsuz bileşenler olarak da adlandırılırlar.
import React from 'react';
import { Text } from 'react-native';
const App = () => {
return (
<Text>Merhaba Dünya</Text>
);
}
export default App;
Sınıf componentleri , React'in Component adlı bir temel sınıfını genişleten JavaScript ES2015 sınıflarıdır. React yaşam döngüsü yöntemlerinin yanı sıra ana sınıftaki state/props işlevselliğine erişim sağlar.
import React, { Component } from 'react';
import { Text } from 'react-native';
class App extends Component {
render() {
return (
<Text>Merhaba Dünya</Text>
);
}
}
export default App;
Çoğu component oluşturulduklarında farklı parametrelerle özelleştirilebilir. Oluşturulan bu parametreler, özelliklerin kısaltması olan prop olarak adlandırılır. Props kullanarak verileri üst görünümden alt görünüme aktarabilirsiniz.
<Button onPress={onPressFunction} title="Learn More" color="#841584" />
//Button component'ni incelersek burada onPress, title ve color props'larını özelleştirdik.
<MyComponent message="Merhaba Dünya" />
//Burada ise custom component olarak oluşturulmuş MyComponent bileşinine message prop'u oluşturarak "Merhaba Dünya" verisini aktardık.
React Native ile uygulamanızı JavaScript kullanarak şekillendirirsiniz. Tüm çekirdek bileşenler stil adında bir prop kabul eder. Stil adları ve değerleri genellikle CSS'in web'de nasıl çalıştığıyla eşleşir, ancak adlar camel casing kullanılarak yazılır, örneğin background-color yerine backgroundColor.
import React from 'react';
import {Text, View } from 'react-native';
const App = () => {
return (
<View style={{marginTop:50}}>
<Text style={{color:"red"}>just red</Text>
</View>
);
};
export default App;
Bir bileşenin karmaşıklığı arttıkça, birkaç stili tek bir yerde tanımlamak için StyleSheet.create kullanmak genellikle daha temizdir.
import React from 'react';
import { StyleSheet, Text, View } from 'react-native';
const App = () => {
return (
<View style={styles.container}>
<Text style={styles.text}>just red</Text>
</View>
);
};
const styles = StyleSheet.create({
container: {
marginTop: 50,
},
text: {
color: 'red',
},
});
export default App;
React ortamında değeri değiştiğinde tanımlaman componenti tekrar render etmesini sağlayan değişkenlere state adı verilir.
React.Component’ini extend ettiği zaman React.Component’nin constructor’ı state’i componentimize tanımlar ve varsayılan değer olarak null atar. Daha sonrasında kendi state değerlerimizi belirlemek için componentimizin constructor’ında veya direkt olarak sınıf içerisinde tanımlamalarımızı gerçekleştirebiliriz. Constructor içerisinde tanımlamış olduğumuz statelere erişmek için this ifadesini kullanırız.
//State Tanımlama
constructor(props) {
super(props);
this.state = { myState: 'Merhaba React Native' };
}
//State değerini değiştirme
this.setState({myState: 'Merhaba Dünya'})
Class componentde React.Component içerisinde state mekanizması hazır olan bir yapıdan bileşenimizi türetiyorken functional component de bunu useState Hook ile sağlayabiliyoruz. useState bir dizi şeklinde tanımlanır iki parametre alır ilk parametre değişkenin (state) kendisi ikincisi değişiklikleri atamamızı sağlayan parametre olarak tanımlanır.
//State Tanımlama
const [myState, setMyState]=useState("Merhaba React Native");
//State değerini değiştirme
setMyState('Merhaba Dünya')
Her componentin bir yaşam süreci vardır. Doğar, yaşar ve ölür. Biz geliştirme sürecinde bu componentlerin yaşam evrelerini lifecycle ile yönetiyoruz.
React Hooks gelmeden önce Class component yapısında kullanılan üç aktif life-cycle method bulunmaktadır bunlar;
componentDidMount()
Bileşen başlangıçta bir kez render edildiğinde çalışır.
componentDidUpdate()
Bileşen update edildiğinde ait olduğu bileşen render edilir ve yenilenir.
componentWillUnmount()
Bileşen yapıdan çıkarıldığında (silinmesi,gösterilmemesi) gibi durumlarda kullanılır .
React Hooks ile birlikte dünyamıza giren useEffect bu yapıları tek bir method altında kullanmamıza olanak sağlıyor.
useEffect (()=>{
},[])
Bileşen başlangıçta bir kez render edildiğinde çalışır.
useEffect (()=>{
},[dependencies])
Bileşen update edildiğinde ait olduğu bileşen render edilir ve yenilenir.
useEffect (()=>{
return()=>{}
},[])
Bileşen yapıdan çıkarıldığında (silinmesi,gösterilmemesi) gibi durumlarda kullanılır .
Veriler prop'lar aracılığıyla en üst componentden bir alt componenete (yukarıdan aşağıya) aktarılır. Ancak bu tür bir kullanım büyük bir uygulama içindeki birçok componenet arası veri aktarımı işlemi için zor ve kullanışsız bir yöntem olacaktır. Context ve Redux vb. yöntemler componentlerin her seviyesinden açıkça bir prop geçirmek zorunda kalmadan bileşenler arasında bu gibi değerleri paylaşmanın bir yolunu sağlar.
React’ın kendi geleneksel context yapısını kullanarak global bir state yönetimi tasarlamaya yöntemidir. Global state değerleri çok fazla değişmeyecekse, birden fazla context yapısı kullanacaksak bu yöntemi tercih edebiliriz.
Redux state bileşenlerini yönetmemizi sağlayan kütüphanedir. Gloal statelerimizde sık sık veri güncellemesi yapacaksak, birden fazla reducer'a ihtiyaç duyacaksak bu yöntemi tercih edebiliriz. Redux’ta veri aktarımı
Action
,Reducer
veStore
gerçekleştirilir ve UI’a sunulur.
-
Action
Uygulama içerisinden store’a iletilen değişkenlerin bilgilerini tutar. -
Reducer
Reducerlar, action sonucunda uygulamanın var olan state’i değiştirmesini sağlar. Uygulama değişikliğinin state’e aktarılması ise reducer tarafından olur. -
Store
Store işlemi, action ve reducerı bir araya getirip yapıyı bağlar. Uygulamanın state’ini tutar ve bazı metodlar ile bu state’ erişim yapılmasını sağlar. -
Provider
Store’un tüm uygulamaya etki etmesini sağlayan, uygulamanın etrafını sarmalayan bir yapıdır.
Redux saga
asenkron akışların okunmasını, yazılmasını ve test edilmesini kolaylaştırmak için kullanılan bir kütüphanedir. Bir Redux ara yazılımıdır, yani bu iş parçacığı normal Redux eylemleriyle ana uygulamadan başlatılabilir, duraklatılabilir ve iptal edilebilir, tüm Redux uygulama durumuna erişebilir ve Redux eylemlerini de gönderebilir.
Bazen uygulamalarımızda fazla işlemci tüketen fonksiyonlar veya gereksiz re-render eden componentlerden kaynaklı performans sorunları yaşayabiliriz. Bu performans sorunlarını önlemek için Class componentler için Pure Component
ve shouldComponentUpdate
, Functional componentler için useMome
ve useCallback
yöntemlerini kullanabiliriz.
Fonksiyonlardaki fazla işlemci tüketen işlemler olduğu durumlarda fonksiyon her çağrıldığında bu işlemleri yapmak yerine fonksiyondan dönen son değeri afızasında tutar. Son değeri (dependency array) referans alarak son değer değişmedikçe cache deki değeri döndüren bir yöntemdir.
Bir component her render edildiğinde component içerisindeki fonksiyonlarda da tekrardan oluşturulur. Bu da büyük projelerde component içerisindeki fonksiyonların tekrar tekar oluşturacağı için performans açısından kötü sonuçlar doğuracaktır. Bu re-render işlemini önlemek için useCallBack Hook'unu kullanıyoruz.
Bir üst component render edildiğinde ona bağlı alt componentlerde render olur. Alt componentlerde yansıyan herhangi bir değişlik yok ise boşa render işlemi gerçekleşmiş olur. React.memo ile sarmaladığımı bir component kendisine gönderilen props değerlerini saklar ve kaydeder. Bir sonraki render durumunda bu component’e gönderilen props değerleri, bir önceki render edildiğindeki props değerleri ile karşılaştırır. Eğer props değerleri aynı ise componenti tekrar render etmez.
Özetle useMemo
fonksiyondan dönen değeri, useCallback
dönen fonksiyonu ve memo
ise componentten dönen prop değerlerini hafızada tutarak gereksiz render işlemlerini önlememize yarıyor.
Projelerimizde kod yapıları genişledikçe beklenmeyen hatalar büyük sorunlara dönüşmektedir. Bir mobil uygulamayı kullanıma sunma süreci web uygulamalarına göre daha uzun bir süre almaktadır. Buglı bir mobil uygulamanın kullanıma sunulduktan sonra güncellemenin gönderilmesi zaman ve maliyet açısından sorun teşkil edecektir. Buda mobil uygulamaların kullanıma sunulmadan testlerinin yapılmasının önemini artırmaktadır. Testler ayrıca projeye yeni katılacak kişiler için kodların işlevselliği için belge görevi görecektir.
Unit
testler bir uygulamada bulunan en küçük yapıların birimlerin test edilmesidir.Integration
testleri birbirinden farklı olan birimlerin bir araya gelerek oluşan yeni yapının doğru bir şekilde çalışıp çalışmadığının kontrol edilmesidir.End to End
veUI
testleri son kullanıcı gibi davranarak uygulamaların tümünün kontrol edildiği testlerdir.
Uygulamalarımızda test edebileceğimiz en küçük birimdir. Bir component üzerinde değişiklik yapılıp yapılmadığını, component üzerinden dönen propsların işlevselliğini, bir fonksiyonun istenen işlevselliği karşılayıp karşılamadığını ve componentlerin stil özelliklerini test edebiliriz.
Jest
: Facebook tarafından test için geliştirilmiş bir kütüphanedir. React ve React-Native gibi frameworklerde kullanabilmekteyiz.
Testing Library
: React dokümanında testler için tavsiye edilen kütüphanedir. Testing library React-Native için yardımcı fonksiyonlar sunar.
Bir component in doğru bir şekilde render olup olmadığının ve component üzerinde değişiklik yapılıp yapılmadığının kontrol edildiği testtir.
import React from 'react';
import {render} fom '@testing-library/react-native';
import Button from './Button';
test('should match with snapshot', ()=>{
const comp=render(<Button/>);
expect(comp).toMatchSnapshot();
})
Component üzerinden dönen propsların işlevselliğinin kontrol edildiği testtir.
import React from 'react';
import {render} fom '@testing-library/react-native';
import Button from './Button';
test('should render title correctly', ()=>{
const testTitle='test';
const comp=render(<Button title={testTitle} />);
const buttonText=comp.getByTestId('button-title').children[0];
expect(buttonText).toBe(testTitle);
})
Bir fonksiyonun istenilen işlevselliği karşılayıp karşılamadığının kontrol edildiği testtir.
import React from 'react';
import {render, fireEvent} fom '@testing-library/react-native';
import Button from './Button';
test('should trigger onPress', ()=>{
const mockFunction=jest.fn();
const comp=render(<Button onClick={mockFunction} />);
const buttonTouchable=comp.getByTestId('button-touchable');
fireEvent(buttonTouchable,'press');
expect(mockFunction).toBeCalled();
})
Componentlerin stil özelliklerinin kontrol edildiği testtir.
import React from 'react';
import {render} fom '@testing-library/react-native';
import Button from './Button';
import styles from './Button.style';
test('should render given theme style', ()=>{
const selectedTheme='primary'
const comp=render(<Button theme={selectedTheme} />);
const buttonStyle=comp.getByTestId('button-touchable').props.style;
expect(buttonStyle).toMatchObject(styles[selectedTheme].container);
})
Entegrasyon testi, farklı parçaların bir grup olarak test edildiği bir test türüdür.
const initialState = {
todos: {
todoList: ['buy groceries'],
},
};
test('should display previous and new todos', async () => {
const newTodoText = 'go running';
const page = renderPage(<TodoList />, initialState);
// GIVEN
const TodoInput = page.getByPlaceholder(wording.todos.newTodo);
const AddTodoButton = page.getByText(wording.todos.add);
const FirstTodo = page.queryByText('buy groceries');
expect(FirstTodo).toBeTruthy();
// WHEN
fireEvent.changeText(TodoInput, newTodoText);
fireEvent.press(AddTodoButton);
// THEN
const NewTodo = await waitForElement(() => page.queryByText(newTodoText));
expect(NewTodo).toBeTruthy();
});
E2E testleri son kullanıcı gibi davranarak uygulamaların tümünün kontrol edildiği testlerdir. E2E testleri yapabileceğimiz çeşitli araçlar mevcuttur. Bunlardan en popüler olanı
Detox
kütüphanesidir. React Native uygulamaları için özel olarak tasarlanmışır. Diğer popüler kütüphaneAppium
'dur.
test('should login successfully', async () => {
await device.reloadReactNative();
await element(by.id('email')).typeText('john@example.com');
await element(by.id('password')).typeText('123456');
await element(by.text('Login')).tap();
await expect(element(by.text('Welcome'))).toBeVisible();
await expect(element(by.id('email'))).toNotExist();
});