Skip to content

VolodymyrPerun/pandagram-react-social-app-main

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

React

##Дисклеймер Всё, что написано далее написано мною лично. Я постарался донести всю информацию максимально понятным языком. Делал я это в первую очередь для себя, чтобы лучше усвоить информацию, так что в некоторых моментах я писал вещи понятные только мне.

Компонента

Компонента - это функция, которая возвращает JSX разметку. Т.е HTML внутри JS. Компоненту никогда не вызывают, Компонента - это ТЕГ.

SPA

SPA - Single page application (Всё происходит в одном HTML файле, который приходит пользователю)

1562585643793

Пропсы

Пропсы - произвольные данные(объекты), которые задаются в виде атрибута ТЕГА(компоненты) и хранятся в объекте под видом Ключ : Значение (например

<Welcome name="Dima" />

имеет вид находясь в объекте, который в свою очередь является параметром функции:

props {
  name : "Dima"
}

Модуль react-router-dom

позволяет использовать route (изначально всё нужно обвернуть в один блок под названием BrowserRouter. Он позволяет отрисовывать компоненты, отслеживания их url ( react <Route path='/profile' component={Profile} /> ​ ). Чтобы задать изменение url нужно использовать тег NavLink (

```react
<NavLink to="/profile">
    Profile
</NavLink>
```

​ )

BLL (Business Logic Layer) - REDUX UI (User Interface) - REACT

Метод для массива MAP

Пример использования: Объявляем массив с объектами, содержащими ключ и значение

let messagesData = [
    {id: 1, message: 'Hello'},
    {id: 2, message: 'Privet'},
    {id: 3, message: 'Aloha'},
];

Создаём новый массив и делаем его равным массиву messagesData с методом map. Параметром будет являться каждый объект массива messagesData. Назовём его просто - d.

let dialogsElement = dialogsData
    .map(d => <DialogItem name={d.name} id={d.id} />);

onClick, ref, VirtualDOM

OnClick

Атрибут OnClick к тегу звучит как "Действие по клику"

<button *onClick*={addPost}>Отправить</button>

Значением этого атрибута может служить анонимная функция, либо же CallBack функция.

CallBack

CallBack функция - это функция обратного вызова. Т.е функция, которая вызывается другой функцией.

Например. Создали функцию:

`let addPost = () => { ` `

`alert(text);`

`}`

Мы можем вызвать её стандартным способом addPost(). Либо же передать её внутрь другой функции

<button *onClick*={addPost}>Отправить</button>

БЕЗ СКОБОК!!!

ref

<textarea *ref*={newPostElement}></textarea>

ref - это ссылка. Это тоже самое, что присвоить элементу id или класс.

Команда ниже аналогична команде

let newPostElement = document.getElementById('id')
let newPostElement = React.createRef();

Но с помощью инструментов React. Это нужно потому, что нельзя обращаться напрямую к DOM.

Взаимодействовать с ссылками можно таким образом:

let text = newPostElement.current.value;

VirtualDOM

React работает с VirtualDOM. Он подгружает компоненты постепенно в DOM, а поэтому мы не можем знать существует ли данный элемент, к которому мы обращаемся, в DOM.

ReRender

1562938967001

Файл render.js:

import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import {addPost} from './redux/state';
import {BrowserRouter} from 'react-router-dom';

export let reRenderEntireTree = (state) => {
  ReactDOM.render(
    <BrowserRouter>
      <App state={state} addPost={addPost} />
    </BrowserRouter>, document.getElementById('root'));
}

Файл state.js:

import {reRenderEntireTree} from "../render";

let state = {...}

export let addPost = (postMessage) => {
  let newPost = {
    id: 3,
    message: postMessage,
  };
  state.profile.postsData.push(newPost);
  reRenderEntireTree(state);
}

export default state;

Файл index.js:

import './index.css';
import * as serviceWorker from './serviceWorker';
import {reRenderEntireTree} from "./render";
import state from './redux/state';



reRenderEntireTree(state);

// If you want your app to work offline and load faster, you can change
// unregister() to register() below. Note this comes with some pitfalls.
// Learn more about service workers: https://bit.ly/CRA-PWA
serviceWorker.unregister();

FLUX Концепция/Архитектура

1562940934753

Концепция звучит так: Изменения в UI мире должны происходить только тогда, когда произойдут изменения в BLL мире

P.S,

Action - действия/изменения,

state inside props - новые данные stat'a передаваемые в UI с помощью пропсов

Инкапсулировать детали - скрыть детали/скрыть логику

Store

Псевдо ООП

import profileReducer from "./profile-reducer";
import messagesReducer from "./messages-reducer";
import friendsReducer from "./friends-reducer";
let store = {
  _state: {
    profile: {
      postsData: [
        { id: 1, message: 'Hi, how are you' },
        { id: 2, message: 'Its my first post' }
      ],
      newPostText : ''
    },
    messages: {
      dialogsData: [
        { id: 1, name: 'Dima' },
        { id: 2, name: 'Julia' },
        { id: 3, name: 'Victor' },
        { id: 4, name: 'Sasha' }
      ],
      messagesData: [
        { id: 1, message: 'Hello' },
        { id: 2, message: 'Privet' },
        { id: 3, message: 'Aloha' }
      ],
      newMessageText : ''
    },
    friends: {
      friendsData: [
        { id: 1, name: 'Dima' },
        { id: 2, name: 'Julia' },
        { id: 3, name: 'Victor' }
      ]
    }
  },
  _callSubscriber() {
    console.log("state changed")
  },

  getState() {
    return this._state;
  },
  subscribe(observer) {
    this._callSubscriber = observer;
  },

  dispatch(action) { //action === actionCreator (EX. addPostActionCreator)

    this._state.profile = profileReducer(this._state.profile, action);
    this._state.messages = messagesReducer(this._state.messages, action);
    this._state.friends = friendsReducer(this._state.friends, action);

    this._callSubscriber(this._state);
  }
}

export default store;

window.store = store;

Перерисовка

Файл index.js

import './index.css';
import * as serviceWorker from './serviceWorker';
import store from './redux/redux-store';
import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import {BrowserRouter} from 'react-router-dom';
let reRenderEntireTree = (state) => {
  ReactDOM.render(
    <BrowserRouter>
      <App state={state} dispatch={store.dispatch.bind(store)} store={store}/>
    </BrowserRouter>, document.getElementById('root'));
}

reRenderEntireTree(store.getState());
store.subscribe(() => {
  let state = store.getState();
  reRenderEntireTree(state);
});

reRenderEntireTree - Функция, которая рисует ui с помощью ReactDOM.render.

В этом же файле мы её вызываем, чтобы при первом запуске страницы увидеть ui

reRenderEntireTree(store.getState());

НО, UI должен перерисовываться каждый раз, когда он изменяется. Для этого мы обращаемся к функции subscribe через store. И передаём ей в качестве параметра reRenderEntireTree со state в параметре.

store.subscribe(() => {
  let state = store.getState();
  reRenderEntireTree(state);
});

Функция subscribe имеет параметр observer, который по выполнению функции должен стать равным другой пустой функции callSubscriber которая в свою очередь уже не становится пустой.

subscribe(observer) {
    this._callSubscriber = observer;
  },

Было:

_callSubscriber() {
    console.log("state changed")
  }

Стало:

_callSubscriber() {
    let state = store.getState();
  	reRenderEntireTree(state);
  }

Как только (например по нажатию кнопки) в dispatch приходит action происходит вызов callSubscriber (отсюда и название функции)

dispatch(action) { //action === actionCreator (EX. addPostActionCreator)

    this._state.profile = profileReducer(this._state.profile, action);
    this._state.messages = messagesReducer(this._state.messages, action);
    this._state.friends = friendsReducer(this._state.friends, action);

    this._callSubscriber(this._state);
  }

Происходит перерисовка UI.

Bind()

Метод bind() создаёт новую функцию, которая при вызове устанавливает в качестве контекста выполнения this предоставленное значение.

Для чего он нужен? Например, если у нас есть функция

_addPost() {let newPost = {id: 3,message: this._state.profile.newPostText,};this._state.profile.postsData.push(newPost);this._state.profile.newPostText ='';this._callSubscriber(this._state);

  },

И мы передаём её в пропсы как callback

addPost={store.addPost}

То при использовании(вызове) мы будем обращаться к ней от имени(внутри объекта) props

props.addPost()

В этом случае все this будут обращаться именно к props. Ближайшему родительскому объекту, а не к _state, как это было задумано. Что вызывает ошибку.

В этой то ситуации мы и используем bind(Объект на который мы хотим сослаться)

addPost={store.addPost.bind(store)

Dispatch, Action, ActionCreator, Reducer

dispatch - это метод(функция), который передаёт action(действие) в reducer.

dispatch(sendMessageActionCreator());

Action - это объект, который содержит минимум type

ActionCreator - это функция, которая создаёт Action.

export let addPostActionCreator = () => ({type: ADD_POST});
export let updateNewPostTextActionCreator = (text) => ({
  type: UPDATE_NEW_POST_TEXT, 
  newText: text});

Также ActionCreator может создавать(возвращать) другие данные.

Reducer - в свою очередь же, это функция, которая принимает state и action

const profileReducer = (state = initialState, action) => {
  switch(action.type) {
    case ADD_POST:
      let newPost = {
        id: 3,
        message: state.newPostText
      };
      state.postsData.push(newPost);
      state.newPostText ='';
      break;
    case UPDATE_NEW_POST_TEXT:
      state.newPostText = action.newText; 
      break;
    default:
      return state;
  }
  return state;
}

Reducer преобразовывает state и возвращает его обратно. Либо возвращает прежний state, если условия не удовлетворились. Reducer проверяет action.type (созданный функцией actionCreator) на соответствие и выполняет логику.

В reducer ДОЛЖЕН передаваться инициализационный state. Иначе функция вернёт undefinned.

let initialState = {
  postsData: [
    { id: 1, message: 'Hi, how are you' },
    { id: 2, message: 'Its my first post' }
  ],
  newPostText : ''
};

Логика такова:

1564063093206

Redux

let store = createStore(reducers);

createStore - это функция, которая принимает в качестве параметра reducer и создаёт store

Либо комплект редьюсоров

let reducers = combineReducers({
    profile : profileReducer,
    messages : messagesReducer,
    friends : friendsReducer
});

Функция combineReducers принимает в качестве параметра объект state : reducer. State в данном случае будет служить родительским объектом initialState. Другими словами initialState - это state, который мы указывает в виде ключа в combineReducers. Каждый reducer будет работать с отдельными state. profileReducer будет изменять profile, messagesReducer - messages и т.д.

Как это было в самописном store:

this._state.profile = profileReducer(this._state.profile, action);
this._state.messages = messagesReducer(this._state.messages, action);
this._state.friends = friendsReducer(this._state.friends, action);

Как это в Redux:

let reducers = combineReducers({
    profile : profileReducer,
    messages : messagesReducer,
    friends : friendsReducer
});

Container Component

Мы используем Контейнерную компоненту, чтобы презентационная компонента не имела связи со Store. Все данные из store будут приходить сначала в оболочку - контейнерную компоненту. Далее эта компонента отрисовывает презентационную компоненту и передаёт ей все данные через props.

Container Component

import React from 'react';
import { sendMessageActionCreator, updateNewMessageTextActionCreator } from '../../redux/messages-reducer';
import Dialogs from './Dialogs';


const DialogsContainer = (props) => {

  let state = props.store.getState();

  let sendMessage = () => {
    props.store.dispatch(sendMessageActionCreator());
  }

  let onMessageChange = (text) => {
    props.store.dispatch(updateNewMessageTextActionCreator(text));
  }
  
  return (
    <Dialogs 
      sendMessage={sendMessage}
      onMessageChange={onMessageChange}
      dialogsData={state.messages.dialogsData}
      messagesData={state.messages.messagesData}
      newMessageText={state.messages.newMessageText}
    />
  );
}

export default DialogsContainer;

Component (Presentation)

import React from 'react';
import s from './Dialogs.module.scss';
import DialogItem from './DialogItem/DialogItem';
import Message from './Message/Message';

const Dialogs = (props) => {
  let dialogsElement = props.dialogsData
    .map(d => <DialogItem name={d.name} id={d.id} />);

  let messagesElement = props.messagesData
    .map(m => <Message message={m.message} id={m.id} />);

  let newMessageElement = React.createRef();

  let onSendMessage = () => {
    props.sendMessage();
  }

  let onMessageChange = () => {
    let text = newMessageElement.current.value;
    props.onMessageChange(text);  
  }
  
  return (
    <div className={s.dialogs}>
      <div className={s.dialogs_items}>
        {dialogsElement}
      </div>
      <div className={s.messages}>
        {messagesElement}
        <div className={s.inputArea}>
          <textarea
            onChange={onMessageChange} 
            ref={newMessageElement} 
            value={props.newMessageText}
            placeholder="Введите сообщение"
          />
          <button onClick={onSendMessage}>Send</button>
        </div>
      </div>
    </div>
  );
}

export default Dialogs;

REACT-REDUX

React-redux это библиотека, позволяющая react работать с redux инкапсулируя все детали.

Давайте вспомним как мы жили раньше, до этой библиотеки.

Context API

Как это было раньше. Мы использовали функцию reRenderEntireTree. Вызывали её в index.js во время старта и также передавали её как callback subscribe'у если требовалось отрисовать дерево заново, если изменялся state.

Index.js

import './index.css';
import * as serviceWorker from './serviceWorker';
import store from './redux/redux-store';
import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import { BrowserRouter } from 'react-router-dom';
import { Provider } from './StoreContext';
let reRenderEntireTree = (state) => {
  ReactDOM.render(
    <BrowserRouter>
      <Provider store={store}>
        <App state={store.getState()}  />
      </Provider>
    </BrowserRouter>, document.getElementById('root'));
}

reRenderEntireTree(store.getState());
store.subscribe(() => {
  let state = store.getState();
  reRenderEntireTree(state);
});

// If you want your app to work offline and load faster, you can change
// unregister() to register() below. Note this comes with some pitfalls.
// Learn more about service workers: https://bit.ly/CRA-PWA
serviceWorker.unregister();

Также мы использовали файл StoreContext для работы с ContextAPI. Мы создавали собственную компоненту Provider внутрь которой передавали store через value с помощью ContextAPI.

И использовали компоненту Provider внутри index.js

Это позволяло давать доступ контейнерным компонентам к store без необходимости прокидывать его через props.

StoreContext.js

import React from 'react';

const StoreContext = React.createContext(null);

export const Provider = (props) => {
  return (
    <StoreContext.Provider value={props.store}>
      {props.children}
    </StoreContext.Provider>
  );
}

export default StoreContext;

ContainerComponent

При создании контейнерной компоненты нам приходилось возвращать разметку внутри StoreContext.Consumer'a. Там же нам приходилось получать state через getState() и dispatch'и. Чтобы всё это передать в презентационную компоненту через пропсы.

DialogsContainer.js

import React from 'react';
import { sendMessageActionCreator, updateNewMessageTextActionCreator } from '../../redux/messages-reducer';
import Dialogs from './Dialogs';
import StoreContext from '../../StoreContext';


const DialogsContainer = (props) => {
  return (
    <StoreContext.Consumer > 
    {
      (store) => {
        let state = store.getState();

        let sendMessage = () => {
          store.dispatch(sendMessageActionCreator());
        }

        let onMessageChange = (text) => {
          store.dispatch(updateNewMessageTextActionCreator(text));
        }
        return (
          <Dialogs
            sendMessage={sendMessage}
            onMessageChange={onMessageChange}
            dialogsData={state.messages.dialogsData}
            messagesData={state.messages.messagesData}
            newMessageText={state.messages.newMessageText}
          />
        )
      }
    }
    </StoreContext.Consumer>
  );
}

export default DialogsContainer;

Давайте же взглянем, как проста стала жизнь!

ContextAPI

  1. Удаляем файл StoreContext.js. Он нам больше не нужен.

  2. Удаляем функцию reRenderEntireTree в index.js! Она нам также больше не нужна.

  3. Компоненту Provider оставляем, но импортируем её уже не через StoreContext, а через react-redux.

import './index.css';
import * as serviceWorker from './serviceWorker';
import store from './redux/redux-store';
import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import { BrowserRouter } from 'react-router-dom';
import { Provider } from 'react-redux';

ReactDOM.render(
  <BrowserRouter>
    <Provider store={store}>
      <App />
    </Provider>
  </BrowserRouter>, document.getElementById('root'));


// If you want your app to work offline and load faster, you can change
// unregister() to register() below. Note this comes with some pitfalls.
// Learn more about service workers: https://bit.ly/CRA-PWA
serviceWorker.unregister();

ContainerComponent

Создание контейнерным компонент ещё не было настолько простым!

  1. const UsersContainer = connect(mapStateToProps, mapDispatchToProps)(Users);

Одна строчка и компонента создана.

Функция connect, импортированная из react-redux, что она делает?

Во-первых, как мы помним, мы удалили функцию reRenderEntireTree для перерисовки и отрисовки дерева. Connect делает всё это сам. И делает намного лучше! Он не перерисовывает всё дерево при изменении в одной компоненте, он перерисовывает только эту компоненту.

Во-вторых, она обеспечивает самый простой способ передачи state и dispatch в презентационную компоненту.

Connect вызывается дважды. В первый раз она вызывается принимая в качестве параметров две функции mapStateToProps и mapDispatchToProps. Их названия говорят за них сами.

mapStateToProps принимает state и возвращает объект ключ-значение в ЗНАЧЕНИИ атрибут-данные.

Проще показать на примере:

Это

<Users users={state.users.usersData} />

Равно этому

let mapStateToProps = (state) => {
    return {
        usersData: state.users.usersData
    }
}

Тоже самое с dispath

mapDispatchToProps принимает dispatch на вход

Это

let follow = (userId) => {
	store.dispatch(followAC(userId));
}

 <Dialogs follow={follow} />

Равно этому

let mapDispatchToProps = (dispatch) => {
	return {
		follow: (userId) => {
			dispatch(followAC(userId));
	},
}

Как работает двойной вызов?

Connect(result=a+b)(result=props.result)

Первый раз функция считает логику в первых скобках, далее результат этих скобок он передаёт во вторые. Второй раз функция выполниться только после выполнения первого раза, и возьмёт на вход результат первого выполнения.

Тоже самое происходит здесь:

const UsersContainer = connect(mapStateToProps, mapDispatchToProps)(Users);

Выполняются функции mapStateToProps и mapDispatchToProps. Они возвращают объекты с данными и функциями. Далее эти функции переходят на вход Users в виде props. Компоненте, которая как мы уже знаем также является функцией.

UsersContainer.jsx

import { connect } from 'react-redux';
import Users from './Users';
import { followAC, unFollowAC, setUsersAC } from '../../redux/users-reducer';

let mapStateToProps = (state) => {
    return {
        usersData: state.users.usersData
    }
}

let mapDispatchToProps = (dispatch) => {
    return {
        follow: (userId) => {
            dispatch(followAC(userId));
        },
        unfollow: (userId) => {
            dispatch(unFollowAC(userId));
        },
        setUsers: (users) => {
            dispatch(setUsersAC(users));
        }
    }
}

const UsersContainer = connect(mapStateToProps, mapDispatchToProps)(Users);

export default UsersContainer;

deep copy vs shallow copy

Пойми одно, это БАЗА. База, которую ты должен знать.

Создадим объект а

let a = {
      name: 'TextA',
      protocol: 'https',
      maxStudents: 10,
      isOnline: true,
      students: ['ivan', 'dima', 'julia', 'andrey'],
      classrom: {
        teacher: {
          name: 'oleg',
          age: 18
        }
      }
    }

создание объекта с помощью синтаксиса

={}

называется созданием объекта с помощью литерала объекта.

Давайте разберёмся в том, как устроен объект.

name: 'TextA',
protocol: 'https',
maxStudents: 10,
isOnline: true,

Данные, хранящие текст(символы), числа, булевые значения, null, undefinned называются примитивами.

students: ['ivan', 'dima', 'julia', 'andrey'],

Ключ, значением которого является массив - на самом деле это ссылка на массив. Массив же, кстати, также является объектом.

Это же касается вложенных объектов типа

classrom: {
        teacher: {
          name: 'oleg',
          age: 18
        }
      }

Ключ classrom является ссылкой на teacher. teacher же является ссылкой на объект, в котором хранятся примитивы name и age.

Почему я называю их ссылками? Всё очень просто. Все объекты, как бы хранящиеся внутри другого объекта на самом деле лежат независимо друг от друга. Они принадлежат разным секторам памяти. Именно поэтому, мы можем использовать их ссылаясь на них.

Всё очень наглядно можно показать на примере.

Мы захотели создать переменную b и приравнять её к объекту a.

let b = a;

Казалось бы, всё готово. И в консоль выводит всё верно.

1564214889509

Два одинаковых массива.

Но НЕТ. Давайте попробуем изменить name в объекте b и вывести в консоль оба объекта. a и b.

b.name = 'TextB'

1564215021643

Как можете видеть, name изменилось и в объекте a.

Всё дело в том, что если мы "копируем" объект таким образом let b = a; То b становится равными ссылке, на которую ссылается a. А именно на объект

{
      name: 'TextA',
      protocol: 'https',
      maxStudents: 10,
      isOnline: true,
      students: ['ivan', 'dima', 'julia', 'andrey'],
      classrom: {
        teacher: {
          name: 'oleg',
          age: 18
        }
      }
    }

Есть способ копирования объекта

let b = {...a}

Он называется поверхностным. И вот почему...

Попробуем опять изменить name в объекте b и выведем консоль

b.name = 'TextB'

1564215283657

Ура! Теперь это на первый взгляд два разных объекта. Но давайте добавим Sema в массив students объекта b. И выведем это

b.students.push('Sema')

1564215428311

Опять всё идёт не по плану. Изменились оба массива.

Как говорилось ранее students - это ссылка на массив. Поверхностное копирование копирует и создаёт новый объект только со значениями первого уровня (первой вложенности). Students же уже вторая.

Её также можно скопировать, но для этого нужно также прописать

b.students = [...a.students]

Это можно делать до бесконечной вложенности.

ВАЖНО

Два объекта НИКОГДА не будут равны, даже если значения внутри них полностью совпадают.

При сравнении примитивов a.name === b.name будут сравниваться именно их значения

При сравнении двух ссылок, (без копирования, просто let b = a). Ссылки будут равны, ибо они ссылаются на один и тот же объект.

При сравнении примитивов внутри ссылок a.classrom.teacher.name === bclassrom.teacher.name будут сравниваться значения примитивов.

Axios, jQuery Ajax

jQuery

Что такое промис? Помню себя, понять не мог ничего. Смотрел уроки... Обещание, это обещание говорили они. Я не понимал НИХЕРА. Что за обещание. Мы о чём. Но вот настал момент когда и я понял что это такое.

Давайте по бумажке, promise - это обещание, которое даёт dal - ui, что возьмёт с сервера данные, взяв их оттуда, он передаёт их в виде того обещания, что он дал ранее.

1564427302517

const resultBlock = document.querySelector('#result');

const clickMeButton = document.querySelector('#click-me');

const pageNumberEl = document.querySelector('#page-number');

clickMeButton.addEventListener('click', () => {
  const promise = getImagesOld();
  promise
    .then((promise) => console.log(promise))
})

function getImagesOld() {
  const promise = $.ajax(`https://api.unsplash.com/photos/random/?client_id=818fcd1d1047300379afae3e5c33cb9cd88b68d7d81253632ba4e09a331246a5&count=30`)
  return promise;
}

const successGet = (data) => {
  data.forEach(el => {
    const img = document.createElement('img');
    img.src = el.urls.thumb;
    document.querySelector('#result').appendChild(img)
  });
}

Синтаксис Ajax jQuery таков:

$.ajax(url, {settings})

settings имеют вид ключ-значение. Самое основное success. Как это звучит:

Как только придёт УСПЕШНЫЙ ответ с сервера выполниться это...

success'у можно передать callback функцию

const promise = $.ajax(`https://api.unsplash.com/photos/random/?client_id=818fcd1d1047300379afae3e5c33cb9cd88b68d7d81253632ba4e09a331246a5&count=30`, {
	success: successFunction
})

Ах да, промисы, мы же про них говорим. Дело в том что строчка:

$.ajax(`https://api.unsplash.com/photos/random/?client_id=818fcd1d1047300379afae3e5c33cb9cd88b68d7d81253632ba4e09a331246a5&count=30`)

Возвращает тот самый промис, это он и есть

function getImagesOld() {
  const promise = $.ajax(`https://api.unsplash.com/photos/random/?client_id=818fcd1d1047300379afae3e5c33cb9cd88b68d7d81253632ba4e09a331246a5&count=30`)
  return promise;
}

GET, POST, PUT, DELETE

Query String Parameters используются только для GET и DELETE

Query String Parameters - это параметры, которые передаются таким образом:

https://website/api/tasks?taskId=123&title=text

То есть сначала идёт endpoint, затем ставится вопросительный знак(?) и передаются параметры:

хттпс://website/?параметр=значение

Если этих параметров несколько, то после каждого значения нужно ставить амперсант (&)

параметр=значение&параметр2=значение2

При использовании POST и PUT запросов нужно использовать тело запроса request payload

axios

function postTasksAxios() {
  const promise = axios.post(APIkeyTasksPost, {
    widgetId : 85429,
    title : 'ZAKAZAKA'
  })
  return promise.then(response => response.data)
}

fetch

async function postTasksFetch () {
  const response = await fetch(APIkeyTasksPost + '&title=WorkOut', {method: 'POST'});
  const data = await response.json();
  return data;
}

Class

class extends React.Component

Что такое класс? Что такое конструктор? Что это за слово такое new?

Для начала давайте вспомним ООП. А именно контекст вызова this. Это слово говорит о том, с чьего имени будет вызываться свойство или метод в объекте.

создадим функцию

function Man2(name, age) {
  this.name = name;
  this.age = age;
}

На что указывает this внутри функции? Ни на что... Пока её не вызвать

Man2('Dimon', 8)

Если просто вызвать данную функцию, то контекстом вызова вызова будет являться глобальный объект window

Это легко можно проверить в консоли браузера после вызова данной функции

1565187371752

Т.е теперь наши name и age - это новые ключи объекта window, а их значения - это параметры функции Man2

Пришло время воспользоваться словом new

создадим переменную и приравняем её вызову функции Man2 напечатав изначально слово new

let m3 = new Man2('Dima', 31);

1565187623311

И мы видим что в консоль вывелся ОБЪЕКТ с КЛАССОМ Man2

В sources это выглядит так

1565187882877

Если бы мы создали этот объект через литерал объекта, то мы бы увидели такую картину

1565187791164

А в sources было бы написано object

1565187836760

Используя слово new можно создавать неограниченное количество ЭКЗЕМПЛЯРОВ класса Man2

Прошу заметить что в данном случае контекстом вызова является переменная к которой мы приравниваем функцию с помощью new

Классы в JavaScript были введены в ECMAScript 2015

Давайте проделаем аналогичные действия с помощью классов

class Man2 {
  constructor(name, age) {
    this.name = name;
    this.age = age;
  }
}

Первое что мы видим что мы видим это то, что в роли нашей функции Man2 играет встроенный метод constructor. Теперь мы знаем как это называется, конструктор. Мы собирали конструктор.

Man2 же ушло немного выше и говорит само за себя, теперь это class.

Ниже конструктора нужно писать все свои методы для работы с данными, которые мы ввели в конструктор

Вот пример кода

class Man {
  constructor(name, age) {
    this.name = name;
    this.age = age;
    this.people = [
      {name: 'Dima', age: 18},
      {name: 'Julia', age: 19}
    ];
    this.Manelement = this.people.map(m => {
      const man = document.createElement('li');
      man.innerHTML = m.name;
      document.querySelector('#resultTask').appendChild(man)
    })
  }

  render() {
    return this.Manelement
  }
}

Мы создали метод render который возвращает Manelement

Чтобы начать им пользоваться, нужно проделать все те же шаги

let m1 = new Man('Dima', 31)

Теперь от имени m1 можно вызвать метод render

m1.render();

Вот и всё.

Для чего нам нужны классы в React'е? А для того, чтобы сохранять чистоту функциональных компонент. Чистая функциональная компонента должна только принимать данные и возвращать ИХ ЖЕ. Она не должна делать запросы на сервер, не должна делать ничего кроме как возвращать JSX разметку

В этом нам и помогает классовая компонента

withRouter

Давайте попробуем разобраться с функцией withRouter, понять как она работает и для чего.

Сначала в коде:

UserItem.jsx

import React from 'react';
import s from './../Users.module.scss';
import userPhoto from '../../../assets/images/768px-Circle-icons-profile.svg.png';
import { NavLink } from 'react-router-dom';

const UserItem = (props) => {

  let follow = () => {
    props.follow(props.id);
  }

  let unfollow = () => {
    props.unfollow(props.id);
  }
  return (
    <NavLink to={'/profile/' + props.id}>
    <div className={s.userItem}>
        <div className={s.userAva}>
          <img src={props.photos.small !== null ? props.photos.small : userPhoto} alt="ava" />
          {props.followed ?
            <button onClick={unfollow}>Unfollow</button>
            :
            <button onClick={follow}>Follow</button>}
        </div>

        <div className={s.userInfo}>
          <h3> {props.fullName} </h3>
          <p className={s.status}> {props.status} </p>
        </div>
    </div>
    </NavLink>
  );
}

export default UserItem;

Мы обернули каждого пользователя на странице Users в тег NavLink, указав путь

to={'/profile/' + props.id}

Мы кликаем на этого пользователя. Далее в файле App.js Route отслеживает изменение url адреса по пути /profile и отрисовывает компоненту ProfileContainer

import React from 'react';
import './App.scss';
import Header from './components/Header/Header';
import Nav from './components/Nav/Nav';
import {Route} from 'react-router-dom';
import DialogsContainer from './components/Dialogs/DialogsContainer';
import UsersContainer from './components/Users/UsersContainer';
import ProfileContainer from './components/Profile/ProfileContainer';

const App = (props) => {
  return (
      <div className='app-wrapper'>
        <Header />
        <Nav />

        <div className='app-wrapper-content'>

          <Route path='/profile' render={ () => <ProfileContainer />} />
          <Route path='/dialogs' render={ () => <DialogsContainer />} />
          <Route path='/users' render={ () => <UsersContainer /> } />

        </div>
      </div>
  );
}

export default App;

Идём внутрь ProfileContainer

import React from 'react';
import Profile from './Profile';
import Axios from 'axios';
import {connect} from 'react-redux';
import {setUserProfile} from '../../redux/profile-reducer';
import { withRouter } from 'react-router-dom';


class ProfileContainer extends React.Component {
  componentDidMount() {
    Axios.get(`https://social-network.samuraijs.com/api/1.0/profile/2`)
      .then(response => {
        this.props.setUserProfile(response.data);
      })
  }

  render() {
    return (
      <Profile {...this.props}
      profileData={this.props.profileData}
      />
    );  
  }
}

const mapStateToProps = (state) => {
  return {
    profileData: state.profile.profileData
  }
};

const mapDispatchToProps = {
  setUserProfile
}

let withRouterProfileContainer = withRouter(ProfileContainer);

export default connect(mapStateToProps, mapDispatchToProps)(withRouterProfileContainer);

Давайте вспомним для чего нужна функция connect. Она принимает в качестве параметров данные из state и action creator, чтобы задиспатчить их в reducer. Осуществляет она это с помощью функций, которые возвращают объект, либо же в случае с dispatch можно использовать готовый объект созданный с помощью литерала объекта. Ибо connect может сам dispatch-ить данные.

В компоненту данные и actioncreators приходят в виде props

connect на выходе выбрасывает новую компоненту

Ну так вот.

Функция withRouter очень похожа на connect. Отличие в том, что connect имеет связь со store, а withRouter имеет связь с url данными.

withRouter принимает в качестве только компоненту, и выбрасывает новую.

Но передаёт в неё свои данные. ТАКЖЕ С ПОМОЩЬЮ props

Ну так вот X2

Попав в данный файл мы видим что здесь задана классовая компонента, НО она не экспортируется. Т.е по сути вызывать будут не её.

Для начала создадим над ней контейнерную компоненту для работы с Router. Назовём её withRouterProfileContainer

let withRouterProfileContainer = withRouter(ProfileContainer);

А далее, используя функцию connect, обернём withRouterProfileContainer в ещё одну компоненту

export default connect(mapStateToProps, mapDispatchToProps)(withRouterProfileContainer);

И вот её то уже и экспортируем по default.

И именно она вызовется в App.js. Давайте взглянем на наши пропсы теперь

1565517952176

Теперь помимо profileData и setUserProfile к нам приходит множество новых данных. Но самое главное находится в match.

  • isExact говорит о точном совпадении адреса указанном в Route и адреса, который прописан в данный момент. False потому, что в App.js указан путь /profile, а у нас сейчас /profile/8

  • path - путь, указанный в app.js

  • url - текущий адрес(Может показаться что они совпадают, но нет, просто наш параметр 8 не определяется)

  • params - объект с параметрами. Сейчас их нет. Чтобы это исправить нужно сделать это:

    <Route path='/profile/:userId?' render={ () => <ProfileContainer />} />

Добавить :userId после слэша. А также ? что будет значить что данный параметр опциональный

Давайте снова глянем на props

1565518489591

Теперь isExact - true и в params есть ключ userId со значением 8

Всё что нам остаётся сделать это достать эту 8ку из пропсов и вставить в API key при get запросе

и поставить условие, если параметр не задан и мы переходим просто по /profile

componentDidMount() {
    let userId = this.props.match.params.userId;
    if (!userId) {
      userId = 2;
    };
    Axios.get(`https://social-network.samuraijs.com/api/1.0/profile/${userId}`)
      .then(response => {
        this.props.setUserProfile(response.data);
      })
  }

Вот и всё!)

1565524485562

Архитектурный паттерн нашего приложения в одной картинке

Начало: 2 уровня. UI - User Interface. BLL - business logic layer.

1565949602035

1565949713701

Эволюция - 3 уровня.UI - User Interface. BLL - business logic layer. DAL - data access layer

1565949848685

Эволюция vol 2.0. Разгрузка UI уровня

1565950092440

1565951914995

Redux-Thunk

users-reducer.js

export const getUsers = (pageSize, currentPage) => {
  return (dispatch) => {
    dispatch(toggleIsFetching(true));

    usersAPI.getUsers(pageSize, currentPage)
      .then(response => {
        dispatch(toggleIsFetching(false));
        dispatch(setUsers(response.items));
        dispatch(setTotalUsersCount(response.totalCount));
      })
  }
}

getUsers здесь является thunkCreator'ом. Он возвращает thunk

Thunk - это функция, которая нужна, чтобы организовать side effect. Например делает запрос на сервер (или же в нашем случаем обращается к dal уровню) и после получения ответа диспатчит action в store.

Это решает проблемы конвейера. Всё начинает идти последовательно.

1565950092440

И BLL уровень становится по истине самым главным. Он становится как бы "менеджером" или "посредником" между всеми уровнями.

Мы также передаём thunkCreator контейнерной компоненте, как мы делали это ранее с actionCreator

const mapDispatchToProps = {
  follow,
  unFollow,
  setTotalUsersCount,
  setCurrentPage,
  getUsers
};

и вызываем

componentDidMount() {
    this.props.getUsers(this.props.pageSize, this.props.currentPage);
  }

Картинка ниже очень хорошо описывает весь процесс

1566127947019

Но есть одно но, наш store умеет принимать только action. Т.е только объект. Как нам быть?

Ну для начала нам конечно же нужно установить redux-thunk плагин

npm i redux-thunk --save

а также вторым параметром функции createStore передать applyMiddleware функцию с параметром thunkMiddleware(на самом деле он экспортируется по дефолту и вы можете назвать его как угодно)

import { createStore, combineReducers, applyMiddleware } from "redux";
import profileReducer from "./profile-reducer";
import messagesReducer from "./messages-reducer";
import friendsReducer from "./friends-reducer";
import usersReducer from "./users-reducer";
import authReducer from "./auth-reducer";
import thunkMiddleware from 'redux-thunk';

let reducers = combineReducers({
    profile : profileReducer,
    messages : messagesReducer,
    friends : friendsReducer,
    users : usersReducer,
    auth : authReducer
});


let store = createStore(reducers, applyMiddleware(thunkMiddleware));

export default store;
window.store = store;

Теперь при попадании thunk в store - store будет знать, что такое thunk. Это функция, а не привычный ему action(объект) и просто вызовет её, передав ей в качестве параметра метод dispatch.

Redirect

Redirect - это стандартная компонента React-router-dom. Она используется (как ни странно) для редиректа.

import {Redirect} from "react-router-dom";

Редирект - это перенаправление на другую страницу. Например, при условии если вы не авторизированы.

В authReucers, в initialstate у нас есть флаг isAuth

let initialState = {
  userId: null,
  email: null,
  login: null,
  isAuth: false,
  isFetching: false
};

Изначально он false. У нас всегда отрисовывается компонента header. Сразу после монтирования у нас идёт запрос на сервер, чтобы получить данные об авторизации. Если мы авторизованы и файлы cookie присутствуют, isAuth изменяется на true. Иначе - остаётся false.

Эти данные мы передаём в ту компоненту, которую мы хотим показать только для того пользователя, который авторизирован.

  if (!props.isAuth) return <Redirect to={'/login'} />;

Делаем проверку и возвращаем Redirect с нужным нам путём.

Конечно же этот путь изначально должен быть прописан в app.js

<Route path='/login' render={ () => <Login /> } />

И должно быть указано то, что мы собираемся отрендерить.

HOC - High Order Component

HOC - High Order Component - это не компонента, как может показаться. Это функция, которая принимает в качестве параметра компоненту, и возвращает новую компоненту. Она как бы создаёт обёртку над принятой компонентой, снабжая её новой логикой - данными.

Давайте применим это на нашей логике с Redirect

До этого нам приходилось в нужную компоненту передавать isAuth из Store и прописывать логику с условием на проверку авторизации. Мы копировали код, вновь и вновь. Это очень плохо, и в этом очень легко запутаться.

Давайте создадим HOC. Назовём её withAuthRedirect

import React from 'react';
import { Redirect } from 'react-router-dom';
import { connect } from 'react-redux';

let mapStateToProps = (state) => {
  return {
    isAuth: state.auth.isAuth
  }
}

export const withAuthRedirect = (Component) => {

  class RedirectComponent extends React.Component {
    render() {
      if (!this.props.isAuth) return <Redirect to={'/login'} />;
      return <Component {...this.props} />
    }
  }

  let ConnectedAuthRedirectComponent = connect(mapStateToProps)(RedirectComponent)


  return ConnectedAuthRedirectComponent
}

Давайте по шагам. Изначально внутри функции withAuthRedirect мы создаём компоненту (Она может быть как классовой, так и функциональной), снабжаем её нужной логикой(в нашем случаем проверкой на авторизованность) и возвращаем пришедшую в нас компоненту. ЗАТЕМ всё это мы оборачиваем connect'ом чтобы передать нашей компоненте isAuth из state.

1566304151424

Мы уже использовали HOC до этого. Например withRouter.

let withRouterProfileContainer = withRouter(AuthRedirectComponent);

withRouter - это функция, которая принимает компоненту и на её основе создаёт новую компоненту но уже с новыми данными.( В нашем случае это данные url). Т.е она берёт старую компоненту, добавляет в неё пропсы и выкидывает под видом новой компоненты.

На HOC withAuthRedirect работает таким же образом. Принимает компоненту, добавляет в неё логику и пропсы и выкидывает под видом новой компоненты.

Compose

Compose - Это функция из библиотеки Redux, которая упрощает 'конвейер', делает его более понятным.

Проще показать на коде

Было

let AuthRedirectComponent = withAuthRedirect(ProfileContainer);

let withRouterProfileContainer = withRouter(AuthRedirectComponent);

export default connect(mapStateToProps, mapDispatchToProps)(withRouterProfileContainer);

Стало

export default compose(
  connect(mapStateToProps, mapDispatchToProps),
  withRouter,
  withAuthRedirect
)(ProfileContainer);

Функция вызывает дважды. Принцип такой же, как с функцией connect. Идём снизу - вверх. Изначально берём ProfileContainer и передаём её в качестве параметра функции withAuthRedirect. Она вызывается и возвращает компоненту. Далее - передаём эту компоненту в качестве параметра функции withRouter. Она вызывается и возвращает очередную новую компоненту. Передаём эту компоненту в качестве второго параметра функции connect(mapStateToProps, mapDispatchToProps). Она вызывается и возвращает НОВУЮ компоненту. Она же (наконец-то) и идёт на экспорт по дефолту.

Теперь мы можем легко вклиниваться в этот конвейерный ад.

componentDidUpdate

componentDidUpdate - это метод жизненного цикла компоненты.

*Сейчас важно вспомнить, что эти методы react вызывает сам, но каждый в разное время и при разных условиях.

componentDidUpdate вызывается сразу же после обновления компоненты. Что мы понимаем под этим обновлением? Это либо:

  • В компоненту пришли новые пропсы
  • В компоненте изменился local state

Данному методу приходят 2 параметра(на самом деле может и 3, но это исключение) это

prevProps - старые пропсы, пропсы до обновления

prevState - старый state.

Давайте разберёмся на практике.

У нас есть компонента профиля пользователя. При переходе на неё мы отправляем 2 асинхронных запроса на сервер. Первый - для получения основных данных пользователя, Второй - для получения только статуса пользователя. Т.к они асинхронны, то мы не можем знать чей ответ мы получим первее.

this.props.getProfile(userId);
this.props.getProfileStatus(userId);

Для отображения статуса я использую параграф <p>. На счёт него переживать не стоит. Пропсы в него рано или поздно придут и пользователь это увидит

<p onDoubleClick={ this.activateEditMode }>{this.props.profileStatus ? this.props.profileStatus : 'изменить статус'}</p>

Но мы же хотим иметь возможность менять статус, верно? Для этого я использую <input />

<input autoFocus={true} 
onBlur={this.deactivateEditMode} 
type="text" 
value={this.state.status}
onChange={this.onStatusChange}/>

value для него берётся из локального state, который в свою очередь берёт данные из пропсов (статус из bll). Так вот, давайте посмотрим что может пойти не так.

  1. Переходим на страницу профиля

  2. Отправляются 2 запроса на сервер.

  3. Так получилось что запрос на получения данных о профиле пришёл РАНЬШЕ.

  4. Отрисовывается компонента за счёт полученных данных.

  5. <p> остаётся пустым (или же просто заглушкой), в локальный state ничего не приходит, значит <input /> неоткуда брать данные, он также остаётся пустым.

  6. Приходят данные по второму запросу(статуса)

  7. В <p> приходят пропсы и отрисовывается статус.

  8. В локальный state приходят данные, НО <input /> уже отрисовался, а значит он не будет вновь брать данные из локального state, он так и останется пустым.

Как данная проблемы решается:

componentDidUpdate(prevProps, prevState) {
    if (prevProps.profileStatus !== this.props.profileStatus) {
      this.setState({
        status: this.props.profileStatus
      })
    }
  }

Этот метод вызовется на 6 шагу. Он как-бы говорит "ага, до этого в пропсах не было статуса, а теперь пришли новые пропсы, и здесь этот статус есть, значит изменю ка я локальный state методом setState, а уже данный метод вызовет новую перерисовку методом render"

После перерисовки локальный state сразу будет иметь статус из пропсов, значит <input /> сможет их оттуда взять.

Redux-Form

Помните как мы воплощали в жизнь FLUX круговорот при каждом нажатии на кнопку на клавиатуре? Помните все эти onChange и onClick? Это очень круто. Правда, благодаря этому мы понимаем суть данной архитектуры.

Мы это делали для разнообразных форм, в которые мы вводили текст. А если этих форм на сайте у нас будет 10?...30?...100? Неужели будет весело копировать один и тот же текст? Засорять глобальный state всеми флажками и временными значениями? НЕТ! Не нужно создавать велосипед, он уже давно создан до нас. Нам остаётся только брать и пользоваться!

Redux-Form это библиотека, которая берёт на себя все обязанности по FLUX круговороту, все формы будут обрабатываться ОДНИМ reducer'ом, созданным данной библиотекой.

Давайте по пунктам:

  1. Устанавливаем пакет Redux-Form

  2. Создаём новую ветку state под названием form и присваиваем ей formReducer, импортированный из пакета redux-form ( В самом пакете он называется reducer, что может ввести в заблуждение, так что импортируйте под названием, которое будет понятно ВАМ, например formReducer)

redux-store.js

import { createStore, combineReducers, applyMiddleware } from "redux";
import profileReducer from "./profile-reducer";
import messagesReducer from "./messages-reducer";
import friendsReducer from "./friends-reducer";
import usersReducer from "./users-reducer";
import authReducer from "./auth-reducer";
import thunkMiddleware from 'redux-thunk';
import { reducer as formReducer } from 'redux-form'

let reducers = combineReducers({
    profile : profileReducer,
    messages : messagesReducer,
    friends : friendsReducer,
    users : usersReducer,
    auth : authReducer,
    form : formReducer
});


let store = createStore(reducers, applyMiddleware(thunkMiddleware));

export default store;
window.store = store;
  1. Для использования redux-form создаёте компоненту
const LoginForm = (props) => {
  return (
    <form onSubmit={props.handleSubmit}>
      <div>
        <Field placeholder="Login" 
        component={Input} 
        name="login"
        validate={[required, maxLength10]} />
      </div>
      <div>
        <Field placeholder="Password" 
        component={Input} 
        name="password"
        validate={[required, maxLength10]} />
      </div>
      <div>
        <Field type="checkbox" 
        component={Input} 
        name="rememberMe"
        /> remember me
      </div>
      <div>
        <button>Login</button>
      </div>
    </form>
  );
}

Всё ДОЛЖНО быть обёрнуто тегом form. ЭТО ВАЖНО

Вместо обычных textarea и input используем компоненты Field, импортированные из redux-from

import { Field, reduxForm } from 'redux-form'

В качестве атрибутов необходимо указать name ( это то, как будет называться ветка подветки login, являющаяся в совю очередь веткой form) Ниже всё поймёте!

А также необходимо указать component. Это то, что будет отрисовывать Field - input, textarea и т.п.

  1. Оборачиваете созданную компоненту HOC'ом под названием reduxForm, также импортированным из redux-form
const LoginReduxForm = reduxForm({
  // a unique name for the form
  form: 'login'
})(LoginForm)

Необходимо указать уникальное имя. ( это то, как будет называться подветка ветки form)

1567066574050

Теперь необходимо отрисовать это

const Login = (props) => {
  return (
    <div className={s.formContainer}>
      <h1>Login</h1>
      <LoginReduxForm/>
    </div>
  );
}

Уже сейчас мы сможем увидеть что наш круговорот автоматически работает

Ставим dubugger в созданной нами форме, и вводим любой символ в поле, например В

1567067268523

Написанное нами значение автоматически внеслось в state. Это можно увидеть по свойству values в объекте login

Вот такие props передаёт HOC в нашу компоненту

1567067454013

  1. Сбор данных с формы по нажатию на кнопку.

Укажите handleSumbit из пропсов как callbak, выполнейшейся по нажатию на кнопку (onSubmit)

    <form onSubmit={props.handleSubmit}>

А в LoginReduxForm в качестве пропсов передайте onSubmit функцию, с помощью которой мы и будем управлять данными с формы.

const Login = (props) => {

  const onSubmit = (formData) => {
    console.log(formData)
  }

  return (
    <div className={s.formContainer}>
      <h1>Login</h1>
      <LoginReduxForm onSubmit={onSubmit}/>
    </div>
  );
}

Ниже на рисунке показано как происходит вызов каждого callback и когда.

reduxform

Redux-Form validators и деструктуризация

Чтобы мы смогли использовать валидаторы для наших Fiel, у нас должна быть написана логика. Логику невозможно запихать внутрь стандартного HTML тега. Потому что это просто ТЕГ, не компонента. Самое первое и логичное что приходит на ум - это создать компоненту, которая будет отрисовывать обычный тег. Так и есть

Для начала создадим сами валидаторы. Валидаторы - это функции, которые принимают введённое в Field значение и следуя логике, написанной нами отдают либо undefinned, либо текст ошибки.

validators.js

export const required = value => {
    if (value) return undefined;
    return "Это поле является обязательным"
}
export const maxLengthCreator = (maxLength) => (value) => {
    if (value.length > maxLength) return `Максимальная длина - ${maxLength} символов`;
    return undefined;
}

Здесь написаны 2 валидатора. Один проверяет есть ли значение (обязательное поле или нет), а другой какова длина введёного нами значения.

Кто вызывает данные функции и куда их нужно передавать? Их нужно передать Field в качестве атрибута validate массивом, в котором через запятую указать нужные валидаторы.

<Field
        name="username"
        type="text"
        component={renderField}
        label="Username"
        validate={[required, maxLength15, minLength2]}
        warn={alphaNumeric}
      />

Компонента Field же сама их и вызовет. Результат, т.е ошибка, если таковая имеется, попадёт в пропсы, которые будет передавать Field компоненте, которую она отрисовывает. Она попадёт в объект meta свойство error.

В случае с максимальной длинной мы должны использовать замыкание, чтобы вводить любое число количества символов. НУ ЛИБО ГОВНОКОДИТЬ. Создавая десятки функций, в которых будут отличаться только цифры.

По-хорошему нам просто нужно создать переменную внутри файла нужной компоненты с формой и приравнять ей вызов данного валидатора.

const maxLength30 = maxLengthCreator(30);

Давайте вернёмся к созданию компоненты, которая возвращает обычный тег

export const FormControl = ({input, meta, ...props}) => {
    const hasError = meta.touched && meta.error;
    return (
        <div className={s.form_control + " " + (hasError ? s.error : "")}>
            <textarea {...input} {...props} ><textarea/>
            { hasError && <span>{meta.error}</span>}
        </div>
    );
}

Создаём переменную hasError (Есть ошибка). Это переменная как бы говорит "ошибка есть, если поле тронуто и есть ошибка".

Далее, если ошибка есть, то классу form_control добавляется класс error

Вот такие стили добавляются

.form_control.error {
  textarea, input {
    border: 2px solid red!important;
  }
  span {
    color: red;
  }
}

Внутри данного div находится textarea, внутрь которой мы должны ОБЯЗАТЕЛЬНО должны передать input, заранее диструктуризированный из пропсов и оставшиеся пропсы. Внутри Input сидят такие вещи как: Имя поля, value и все стандартные методы и функции по взаимодействию с тегом типа onChange, onFocus и другие. Стандартный html тег не умеет обращаться сначала к пропсам, чтобы достать оттуда Input, а уже оттуда например значение. Вот поэтому мы и применяем деструктуризацию.

Под нашим полем для ввода мы оставляем span, который виден только при наличии ошибки. Внутрь этого span мы передаём сам текст ошибки. Как мы уже знаем, он находится в meta.error.

Теперь рубрика "Нет говнокоду!"

Что если вместо textarea мы ходим использовать input? Копировать код? ДА ТЫ ГОВНОКОДЕР. Сам задумайся, отличаться будет только одна строчка.

Давай лучше сделаем так

import React from 'react';
import s from './FormsControls.module.scss';

export const FormControl = ({input, meta, ...props}) => {
    const hasError = meta.touched && meta.error;
    return (
        <div className={s.form_control + " " + (hasError ? s.error : "")}>
            {props.children}
            { hasError && <span>{meta.error}</span>}
        </div>
    );
}

export const Textarea = (props) => {
    const {input, ...restProps} = props;
    return <FormControl {...props} >
        <textarea {...input} {...restProps} ></textarea>
    </FormControl>
}
export const Input = (props) => {
    const {input, ...restProps} = props;
    return <FormControl {...props} >
        <input {...input} {...restProps} ></input>
    </FormControl>
}

Композиция, что это и зачем.

Давайте покажу на практическом примере что такое композиция

export const FormControl = ({input, meta, ...props}) => {
    const hasError = meta.touched && meta.error;
    return (
        <div className={s.form_control + " " + (hasError ? s.error : "")}>
            {props.children}
            { hasError && <span>{meta.error}</span>}
        </div>
    );
}

export const Textarea = (props) => {
    debugger
    const {input, ...restProps} = props;
    return <FormControl {...props} >
        <textarea {...input} {...restProps} ></textarea>
    </FormControl>
}

Мы пишем тег, и внутрь него помещаем нужный нам контент(например другую компоненту)

<FormControl {...props} >
    <textarea {...input} {...restProps} ></textarea>
</FormControl>

Чтобы до неё достучаться, а именно указать место, где она будет выводиться, нужно обратиться к children внутри props

<div className={s.form_control + " " + (hasError ? s.error : "")}>
    {props.children}
	{ hasError && <span>{meta.error}</span>}
 </div>

Давайте покажу как это выглядит внутри

export const Textarea = (props) => {
    const {input, ...restProps} = props;
    const hasError = meta.touched && meta.error;
    return (
        <div className={s.form_control + " " + (hasError ? s.error : "")}>
            <textarea {...input} {...restProps} ></textarea>
            { hasError && <span>{meta.error}</span>}
        </div>
    );
}

Композиция необходима, в первую очередь для рефакторинга кода. Дабы избежать повторного кода.

Ещё пару слов про деструктуризацию. Что это и с чем едят.

Она строится на базе rest оператора (...)

Проще показать на практике

let log = (prop) => {
	console.log(prop)
}

log(1,2,3,4,5);

1567363368480

А теперь применим rest оператор

let log = (...prop) => {
	console.log(prop)
}

log(1,2,3,4,5);

1567363461202

Он объединил все входные параметры в одну сущность.

Давайте теперь передадим этой функции объект

const prop = {
    input: {
        name: 'Dima',
        lastName: 'Osincev',
        love: 'Julia'
    },
    meta: {
        sky: 'Blue',
        sun: 'Yelow'
    },
    type: null
}

let log = (prop) => {
	console.log(prop)
}

log(prop);

1567363747497

Выводит объект. Если применить rest оператор на параметр, то он выведет это

1567363862457

Теперь давайте попробуем это

let log = ({input, ...props}) => {
	console.log(input)
}

log(prop);

1567364011979

Выводит только input. Все же остальные свойства объединяются в одну сущность под общим названием props (Можно называть как угодно)

Давайте попробуем вывести эти props

let log = ({input, ...props}) => {
	console.log(props)
}

log(prop);

1567364125842

Reselect

P.S Сознаюсь, вот и настал тот момент когда я просто взял информацию с medium. Мне очень понравилось их донесение, поэтому я не придумал ничего лучше, чем оставить их посыл таким, какой он есть.

Перед тем, как говорить о библиокете reselect, давайте разберём что такое селекторы

Селекторы - это геттеры для состояния редукции. Как и геттеры, селекторы инкапсулируют структуру состояния и могут повторно использоваться. Селекторы также могут вычислять производные свойства.

Пример селектора:

const requestUsers = (state) => {
    return state.users.usersData;
}

Использование:

const mapStateToProps = (state) => {
  return {
    usersData: requestUsers(state)
  }
};

При совместном использовании React и Redux представляют собой потрясающую комбинацию технологий, помогающих нам структурировать приложения с настоящим разделением задач. Даже при том, что React чрезвычайно эффективен из коробки, наступает время, когда требуется еще большая производительность.

Одной из наиболее дорогостоящих операций, которые может выполнять React, является цикл рендеринга. Он запускается, когда компонент обнаруживает изменение входных данных.

Когда мы впервые начинаем работу с React, мы обычно не беспокоимся о том, насколько затратны циклы рендеринга. Но по мере усложнения наших пользовательских интерфейсов нам требуется об этом задумываться. React предлагает нам некоторые инструменты для захвата цикла рендеринга и предотвращения повторной перерисовки, если мы сочтем, что в этом нет необходимости. Для этого мы можем воспользоваться событием жизненного цикла — componentShouldUpdate, которое возвращает boolean, отвечающий за то, будет ли компонент обновлен или нет. Это основа PureRenderMixin, который сравнивает входящие свойства (props) и состояние (state) с предыдущими свойствами и состоянием и возвращает false, если они равны.

Это, в сочетании с неизменяемыми наборами данных, дает нам существенное улучшение производительности, поскольку мы можем легко определить, должен ли компонент повторно отрисовываться или нет. К сожалению, и этого не достаточно.

Рассмотрим следующую проблему. Мы создаем корзину покупок с тремя типами входящих данных:

  • Товары в корзине
  • Количество товаров
  • Налог (зависящий от региона)

Проблема состоит в том, что всякий раз, когда изменяется состояние любого из пунктов (добавляется новый элемент, изменяется количество или изменяется состояние выбора), все нужно будет пересчитать и повторно отрисовать. Вы можете увидеть, как это будет проблематично, если в нашей корзине есть сотни предметов. Изменение процента налога приведет к пересчету позиций в корзине, но не должно. Процент налога — это просто изменение в полученных данных. Только общая сумма и общая сумма налога должны меняться и запускать последующую перерисовку. Давайте посмотрим, как мы можем исправить эти проблемы.

Reselect — это библиотека для создания мемоизированных селекторов (memoized selectors). Мы определяем селекторы как функции, извлекающие фрагменты состояния Redux для наших компонентов React. Используя мемоизацию, мы можем предотвратить ненужные перерисовки и пересчеты полученных данных, что, в свою очередь, ускорит наше приложение.

Если бы у нас было несколько сотен или тысяч вещей, перерисовка всех предметов в нашей корзине была бы дорогостоящей, даже если бы менялся только процент налога. А если бы мы реализовали поиск? Должны ли мы повторно пересчитывать все элементы и налоги каждый раз, когда пользователь ищет что-то в корзине? Мы можем предотвратить эти дорогостоящие операции, перемещая их использование в мемоизированные селекторы. При использовании мемоизированных селекторов, если дерево состояний велико, нам не нужно беспокоиться о том, что дорогие вычисления выполняются каждый раз при изменении состояния. Мы также можем добавить дополнительную гибкость для нашего интерфейса, разбив их на отдельные компоненты.

Давайте посмотрим на простой селектор, используя Reselect:

444444444

В приведенном выше примере, мы разбили нашу функцию поиска товаров в корзине на две функции. Первая функция (строка 3) просто получит все элементы в корзине, а вторая функция является мемоизированным селектором. Reselect предоставляет createSelectorAPI, позволяющий нам создать мемоизированный селектор. Это означает, что getItemsWithTotals будет вычисляться при первом запуске функции. Если эта же функция вызывается снова, но входные данные (результат getItems) не изменились, функция просто вернет кешированный расчет элементов и их итогов. Если элементы изменены (например, добавлен элемент, изменилось количество, любые манипуляции с результатом getItems), функция снова будет выполнена.

Это мощная концепция, позволяющая нам полностью оптимизировать те компоненты, которые должны быть перерисованы, и когда их производное состояние должно быть пересчитано. Это означает, что нам больше не нужно беспокоиться о getItems: общая стоимость каждого элемента начинает рассчитываться, когда операции не зависят от изменений состояния.

DOM

DOM - Document Object Model. Объектная модель документа. Создание для каждого тега на странице отдельного объекта

VirtualDom. Он нужен для того, чтобы создавать виртуальную версию DOM и сравнивать её с настоящим DOM для дальнейшей ТОЧЕЧНОЙ замены объекта.

Помните как создавать элементы на странице с помощью нативного JS?Нет? Сейчас напомню...

const h1 = document.createElement("h1");
document.querySelector("body").appendChild(h1);

Этот код создаёт h1 заголовок и вставляет его в body.

DOM изменяется - страница перерисовывается. Что-то напоминает да? Почему бы тогда всё так и не оставить? Вот пример

const items = [1,2,3,4,5];
items.map( e => {
	let item = document.createElement('p');
	item.innerHTML = e
	document.querySelector('body').appendChild(item)
})

Отрисовка коллекции элементов из массива. Дело в том, что если мы попытаемся изменить хотя бы один элемент в массиве, то Да, страница перерисуется. Но нативному js абсолютно плевать ЧТО-Именно изменилось и он перерисует страницу ПОЛНОСТЬЮ. Т.е перерисовка на каждое действие. На каждую манипуляцию с DOM деревом. Это очень дорого и неоптимизированно.

Тут нам и помогает VirtualDOM. А именно процесс Reconciliation. Согласование на русском.

Что такое компонента? Компонента - это функция или класс, который принимает пропсы и возвращает jsx разметку. Так то оно так, но дело в том, что браузер не знает что такое jsx. Больше скажу, очень часто он не знает обычного js нового стандарта. Всё это конечно зависит от браузера. Если в самые трендовые браузеры, такие как хром или firefox постоянно добавляют поддержку новых фич. То такие браузеры как IE или Safari способны переваривать только код, который был актуален несколько лет назад. Для решения проблем несовместимости был создан Babel.

Babel - это js компилятор, который в основном используется для преобразования кода ECMAScript 2015+ в обратно совместимую версию JavaScript, которая может быть запущена более старыми механизмами JavaScript.

Его работу можно увидеть своими глазами введи в гугле babel online

Ну к чему это я. Babel также нужен для компиляции jsx разметки в обычный js код.

Вам напоминает что-нибудь эта строчка?

React.createElement("div");

Это создание элемента для virtualDom. Синтаксис почти такой же как и для DOM.

Теперь давайте заглянем под капот и вы сразу всё поймёте.

image-20191023235920585

Так мы можем понять, что на самом деле компонента возвращает не jsx разметку, а обычный js код.

Так же, теперь мы понимаем зачем импортировать React в каждый файл, где есть компонента. Всё потому, что мы создаём элемент от лица React. Что позволяет нам создавать его в VirtualDom.

Reconciliation

Это процесс сравнения объектом VirtualDom с объектами DOM и замена изменившихся элементов. Это позволяет нам не перерисовывать компоненту полностью, а точечно изменять объекты.

Именно для этого мы и должны использовать key. Чтобы реакт при сравнении элементов смотрел на их уникальные ключи и перерисовывал всё правильно.

Pure Function

Чиста функция - это функция, обладающая следующими свойствами:

  1. immutability (имьютабельность, неизменяемость) - входные данные, пришедшие в функцию, эта функция не должна менять (речь про объекты и массивы, так как по ссылке они передаются, поэтому делаем копию)
  2. отсутствие side-effects (пункт 1 связан с этим, а так же использование глобальных переменных, их модификация, асинхронные операции и что-то может быть ещё)
  3. детерменированность\идемпотентность - сколько бы раз на вход чистой функции не подавали одно и тоже, на выходе чистой функции должен быть один и тот же результат
  4. чистая функция должна вернуть (return) что-либо

About

react redux learning pet project

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published