zoukankan      html  css  js  c++  java
  • [React] Cleaning up Functional Components with Custom Hooks

    In this lesson we take all of the hooks and logic from our <ExchangeRate> component and put them into a custom hook called useCurrencyCodes().

    What's the difference between a custom hook and a function we can use to organize our logic? Not much, really. Because custom hooks usually execute other hooks, they cannot be called conditionally (meaning we can't say if (ready) useCustomHook()). They must be called in the component's function body or in another custom hook. Redux hooks and React's built-in hooks are both included in our custom hook and work seamlessly together.

    Not only is it convenient to organize your logic in custom hooks, it makes them more sharable and testable as well. I don't recommend putting all of your logic into one large hook, but instead, consider where it would make your component easier to read by combining related hooks and logic into their own function.

    For more details about this pattern please check out this article from the React docs: https://reactjs.org/docs/hooks-custom.html

    Old:

    import React from "react";
    import PropTypes from "prop-types";
    import { connect } from "react-redux";
    import { ratesUpdated } from "../store/actions/RateActions";
    import {
      getCurrencyCode,
      getSupportedCurrencies,
    } from "../store/reducers/RateReducer";
    import { RateTableContainer } from "./RateTable";
    import { CurrencyCodePickerContainer } from "./CurrencyCodePicker";
    import { getExchangeRates } from "../api";
    import { AmountFieldContainer } from "./AmountField";
    
    export class ExchangeRate extends React.Component {
      constructor(props) {
        super(props);
        this.getLatestExchangeRates();
      }
      componentDidUpdate(prevProps) {
        if (this.props.currencyCode !== prevProps.currencyCode) {
          this.getLatestExchangeRates();
        }
      }
      getLatestExchangeRates() {
        const { currencyCode, updateRates, supportedCurrencies } = this.props;
        getExchangeRates(currencyCode, supportedCurrencies).then((rates) => {
          updateRates(rates);
        });
      }
      render() {
        return (
          <>
            <section>
              <h1 className="ExchangeRate-header">
                Exchange Rates <CurrencyCodePickerContainer />
              </h1>
            </section>
            <section>
              <AmountFieldContainer />
            </section>
            <section>
              <RateTableContainer />
            </section>
          </>
        );
      }
    }
    
    // props types
    ExchangeRate.propTypes = {
      updateCurrencyCode: PropTypes.func,
      currencyCode: PropTypes.string,
      supportedCurrencies: PropTypes.arrayOf(PropTypes.string),
    };
    
    // redux stuff
    function mapStateToProps(state) {
      return {
        supportedCurrencies: getSupportedCurrencies(state),
        currencyCode: getCurrencyCode(state),
      };
    }
    function mapDispatchToProps(dispatch) {
      return {
        updateRates: (rates) => dispatch(ratesUpdated(rates)),
      };
    }
    export const ExchangeRateContainer = connect(
      mapStateToProps,
      mapDispatchToProps
    )(ExchangeRate);

    New:

    import React, { useEffect } from "react";
    import { useSelector, useDispatch } from "react-redux";
    import { ratesUpdated } from "../store/actions/RateActions";
    import {
      getCurrencyCode,
      getSupportedCurrencies,
    } from "../store/reducers/RateReducer";
    import { RateTable } from "./RateTable";
    import { CurrencyCodePicker } from "./CurrencyCodePicker";
    import { getExchangeRates } from "../api";
    import { AmountField } from "./AmountField";
    
    export function ExchangeRate() {
      useCurrencyCodes();
      return (
        <>
          <section>
            <h1 className="ExchangeRate-header">
              Exchange Rates <CurrencyCodePicker />
            </h1>
          </section>
          <section>
            <AmountField />
          </section>
          <section>
            <RateTable />
          </section>
        </>
      );
    }
    
    function useCurrencyCodes() {
      const dispatch = useDispatch();
      const supportedCurrencies = useSelector(getSupportedCurrencies);
      const currencyCode = useSelector(getCurrencyCode);
      const updateRates = (rates) => dispatch(ratesUpdated(rates));
      useEffect(() => {
        getLatestExchangeRates();
      }, [currencyCode]);
      function getLatestExchangeRates() {
        getExchangeRates(currencyCode, supportedCurrencies).then((rates) => {
          updateRates(rates);
        });
      }
    }
  • 相关阅读:
    UIScrollView的delegate方法妙用之让UICollectionView滑动到某个你想要的位置
    RSA算法及其在iOS中的使用
    利用UICollectionViewFlowLayout的隐式动画实现UICollectionView的layout的动画调整(外加放大指定cell效果)
    给iOS开发新手送点福利,简述UIPageControl的属性和用法
    给iOS开发新手送点福利,简述UISegment的属性和用法
    给iOS开发新手送点福利,简述UIView的属性和用法
    给iOS开发新手送点福利,简述UIControl事件的用法
    给iOS开发新手送点福利,简述UIAlertView的属性和用法
    给iOS开发新手送点福利,简述UIActivityIndicatorView的属性和用法
    给iOS开发新手送点福利,简述UITextField的属性和用法
  • 原文地址:https://www.cnblogs.com/Answer1215/p/15253016.html
Copyright © 2011-2022 走看看