Press → to see first slide Press ESC to see all slides
data:image/s3,"s3://crabby-images/80e0f/80e0f5e138aeba5f7ce0dc5b5d0cc0ece3a2cf34" alt=""
Jakub Sowiński
3 lata życia aplikacji napisanej w React i TypeScript: lekcje i wnioski
data:image/s3,"s3://crabby-images/f1f79/f1f793561ab8c2f4543427903d21f608013a8824" alt=""
3 lata życia
aplikacji napisanej
w React i TypeScript:
lekcje i wnioski
🗺️
👉 Założenia 📌
👉 Implementacja
👉 Życie na produkcji
🙋?
Problemy
oryginalnej aplikacji:
🔥 nie działa
⚠️ niejasny data flow
data:image/s3,"s3://crabby-images/6a3b0/6a3b0dcd9ed3a59c75f54714e2cf733e0aeaf4be" alt=""
Problemy
oryginalnej aplikacji:
🔥 nie działa
⚠️ niejasny data flow
⚠️ słaby performance
data:image/s3,"s3://crabby-images/c372c/c372c05a37e30033be0d92c558278da7b459b4f6" alt=""
Problemy
oryginalnej aplikacji:
🔥 nie działa
⚠️ niejasny data flow
⚠️ słaby performance
⚠️ brak możliwości rozbudowy
data:image/s3,"s3://crabby-images/9594d/9594dacec89838c1039c11eb22f9a94424a4bdb0" alt=""
Założenia rozwiązania
👌 separacja kodu na moduły
data:image/s3,"s3://crabby-images/1be4b/1be4b8db24d335c22c2b95b0312ddf7885e2bbce" alt=""
Założenia rozwiązania
👌 separacja kodu na moduły
👌 deterministyczny data flow
data:image/s3,"s3://crabby-images/66ef6/66ef61b10462e7ceaf32387806be8d13118bb0dc" alt=""
Założenia rozwiązania
👌 separacja kodu na moduły
👌 deterministyczny data flow
👌 otwartość na modyfikacje
👌 lepszy performance
🗺️
👉 Założenia
👉 Implementacja 📌
👉 Życie na produkcji
data:image/s3,"s3://crabby-images/a2f3d/a2f3df95454900fbc30357ae8e81e3c075a276d5" alt=""
React
data:image/s3,"s3://crabby-images/f9262/f92629f2a6d2eba222185e8e966df7312314320a" alt=""
data:image/s3,"s3://crabby-images/28fc3/28fc30183e709cf39d7ec0586a25edce40cbfc21" alt=""
/src
/components
/PersonalDetailsEdit
/PersonalDetailsPreview
...
/containers
/App
/PersonalDetails
...
src/containers/App/index.tsx
export class App extends React.Component {
render() {
return(
<PageLayout loading={this.props.loading}>
<PersonalDetails />
<MediaQuery maxWidth={this.props.screenMdMax}>
<ProfileCompleteness />
<MatchMyCv />
</MediaQuery>
<Skills />
<Languages />
<WorkExperience />
<Education />
<WorkPreferences />
{this.props.profileVisibilityEnabled
&& <ProfileVisibility />}
<Toast />
<ErrorHandler />
</PageLayout>
);
}
}
src/containers/PersonalDetails/index.tsx
export class PersonalDetails extends React.Component {
render() {
return(
<SectionWrapper sectionName={this.props.sectionName} >
<SectionBody>
{this.props.editMode
? <PersonalDetailsEdit
data={this.props.data}
/>
: <PersonalDetailsPreview
data={this.props.data}
/>
}
</SectionBody>
</SectionWrapper>
);
}
}
data:image/s3,"s3://crabby-images/ea76f/ea76fcec9a2a86f5609c51caa331e8e459adf5b9" alt=""
Redux
/src
/components
/PersonalDetailsEdit
/PersonalDetailsPreview
...
/containers
/App
/PersonalDetails
...
/store
/personalDetails
...
src/store/personalDetails/reducer.tsx
import { PERSONAL_DETAILS_SAVE } from 'store/actionTypes';
export const initialState = { loading: false };
export const appReducer = (state = initialState, action) => {
switch (action.type) {
case PERSONAL_DETAILS_SAVE:
return {
loading: true,
...state,
}
default:
return { ...state };
}
};
src/containers/PersonalDetails/index.tsx
import { connect } from 'react-redux';
import { onPersonalDetailsSave } from './actions'
export class PersonalDetails extends React.Component { /* ... */ }
const mapStateToProps = (state) => ({
data: state.personalDetails.data,
editMode: state.personalDetails.editMode,
});
const mapDispatchToProps = (dispatch) => ({
onSave: () => dispatch(onPersonalDetailsSave),
})
export const PersonalDetailsContainer = connect(
mapStateToProps,
mapDispatchToProps
)(PersonalDetails);
Globalny stan
data:image/s3,"s3://crabby-images/e9229/e9229697f72210c47d084afb042f6c99ccc9598c" alt=""
Redux Dev Tools
data:image/s3,"s3://crabby-images/596cb/596cb3f26873e1f3fcd3a579374c14b897bdd630" alt=""
data:image/s3,"s3://crabby-images/96f3c/96f3cf4cb1aac174ad2d522201b6d3584c9cc402" alt=""
Redux-Saga
src/containers/PersonalDetails/saga.ts
import { put, call } from 'redux-saga/effects';
import { takeLatest } from 'redux-saga';
import { PERSONAL_DETAILS_SAVE } from 'store/actionTypes';
import {
updatePersonalDetailsRequestSuccess,
updatePersonalDetailsRequestFailure,
} from './actions';
function* updatePersonalDetails(action) {
const response = yield call(
fetch,
'/public-api/v1/profile/personal-details',
{ body: JSON.stringify(action.data) },
);
yield put(response.err
? updatePersonalDetailsRequestFailure(response)
: updatePersonalDetailsRequestSuccess(response));
}
export function* watchPersonalDetails() {
yield takeLatest(PERSONAL_DETAILS_SAVE, updatePersonalDetails);
}
data:image/s3,"s3://crabby-images/60e62/60e62525cbde9a75f86aa71c8efb2b3c25801f22" alt=""
Redux Form
data:image/s3,"s3://crabby-images/f76b5/f76b53f595497ad67b6c22a4516365ebe733af99" alt=""
data:image/s3,"s3://crabby-images/1b80a/1b80a2d39a34c83da8a7f665815446945b0eb8c6" alt=""
data:image/s3,"s3://crabby-images/9e38a/9e38aa4ec97a88ff111b42e8200d4780fe2c0150" alt=""
TypeScript
data:image/s3,"s3://crabby-images/41f4f/41f4fc1d93d031341b18ab9154de13f82e3b9b0e" alt=""
data:image/s3,"s3://crabby-images/c4b1d/c4b1ddd9892c8842fe9b76bd90fec3fe4e403a2a" alt=""
💀 Szkielet aplikacji
🗺️
👉 Założenia
👉 Implementacja
👉 Życie na produkcji 📌
😱
🐛 Wykrywanie błędów
data:image/s3,"s3://crabby-images/d1b6e/d1b6ee524071677392fd9bd825a548de3b3ad788" alt=""
data:image/s3,"s3://crabby-images/d521c/d521c87a57805f43b03b1a853e2d39f033ea3919" alt=""
data:image/s3,"s3://crabby-images/39e1a/39e1aa6865a0decac4b011b73fa554450c4f878a" alt=""
Property based tests
import { sum } from '../';
import fc from 'fast-check';
describe('sum function', () => {
test('checks adding two random numbers', () => {
fc.assert(
fc.property(fc.float(), fc.float(), (a, b) => {
expect(sum(a, b)).toBe(a + b);
}),
);
});
});
🚒 Rozszerzanie aplikacji
src/containers/App/index.tsx
export class App extends React.Component {
render() {
return(
<PageLayout loading={this.props.loading}>
<NewSection />
/* ... */
</PageLayout>
);
}
}
webpack.config.js
entry: {
app: 'src/containers/App/index.tsx',
altApp: 'src/containers/AltApp/index.tsx',
},
data:image/s3,"s3://crabby-images/33343/33343a057caaf7c8d3024c59f08aae4524e9e573" alt=""
Założenia rozwiązania
✔️ separacja kodu na moduły
✔️ deterministyczny data flow
✔️ otwartość na modyfikacje
❌ lepszy performance
data:image/s3,"s3://crabby-images/c372c/c372c05a37e30033be0d92c558278da7b459b4f6" alt=""
data:image/s3,"s3://crabby-images/f6193/f6193f1dff47c559350799945cc3492fca6584e9" alt=""
💡
Thank you
Press ← to see last slide Press ESC to see all slides