From 7db93e78331950a95821e724ec3a000b00af40c4 Mon Sep 17 00:00:00 2001 From: "Taehwan, No" Date: Sun, 21 Jan 2018 20:36:28 +0900 Subject: [PATCH] Change pagination UX to show current and max page number (#21) * Show current page number in Pagination component * Change loading indicator UI/UX to rely on fetching state In the previous implementation, show a loading indicator infinitely when API server response has no items. So, I improve this. --- .../HackerNewsList/HackerNewsList.scss | 4 ++ .../HackerNewsList/HackerNewsList.spec.jsx | 5 ++ .../HackerNewsList/HackerNewsList.stories.jsx | 4 +- .../HackerNewsList.spec.jsx.snap | 7 ++- app/components/HackerNewsList/index.jsx | 11 +++- app/components/Pagination/Pagination.spec.jsx | 20 ++++-- .../__snapshots__/Pagination.spec.jsx.snap | 5 ++ app/components/Pagination/index.jsx | 11 +++- app/containers/HackerNewsListContainer.jsx | 3 +- .../__tests__/reducers/isFetching.spec.js | 62 +++++++++++++++++++ app/store/__tests__/selectors.spec.js | 5 ++ app/store/isFetching/index.js | 21 +++++++ app/store/rootReducer.js | 2 + app/store/selectors.js | 1 + 14 files changed, 150 insertions(+), 11 deletions(-) create mode 100644 app/store/__tests__/reducers/isFetching.spec.js create mode 100644 app/store/isFetching/index.js diff --git a/app/components/HackerNewsList/HackerNewsList.scss b/app/components/HackerNewsList/HackerNewsList.scss index 454f320..afc075a 100644 --- a/app/components/HackerNewsList/HackerNewsList.scss +++ b/app/components/HackerNewsList/HackerNewsList.scss @@ -6,6 +6,10 @@ } } +.HackerNewsList__noti { + text-align: center; +} + .HackerNewsList__item { margin: 30px 0; } diff --git a/app/components/HackerNewsList/HackerNewsList.spec.jsx b/app/components/HackerNewsList/HackerNewsList.spec.jsx index 39d4545..54ae352 100644 --- a/app/components/HackerNewsList/HackerNewsList.spec.jsx +++ b/app/components/HackerNewsList/HackerNewsList.spec.jsx @@ -40,4 +40,9 @@ describe('', () => { const wrapper = shallow(); expect(wrapper).toMatchSnapshot(); }); + + it('should have .HackerNewsList__noti when have no items', () => { + const wrapper = shallow(); + expect(wrapper.find('.HackerNewsList__noti')).toHaveLength(1); + }); }); diff --git a/app/components/HackerNewsList/HackerNewsList.stories.jsx b/app/components/HackerNewsList/HackerNewsList.stories.jsx index 6d6455a..cf1b07d 100644 --- a/app/components/HackerNewsList/HackerNewsList.stories.jsx +++ b/app/components/HackerNewsList/HackerNewsList.stories.jsx @@ -13,7 +13,9 @@ stories {story()} )) - .add('default', () => ( + .add('fetching data', () => ) + .add('no data', () => ) + .add('with data', () => ( should match snapshot when render default 1`] = ` className="HackerNewsList" > should match snapshot when render default 1`] = ` } } /> +

+ There are no items to show. +

`; diff --git a/app/components/HackerNewsList/index.jsx b/app/components/HackerNewsList/index.jsx index fdebf35..6b20bc8 100644 --- a/app/components/HackerNewsList/index.jsx +++ b/app/components/HackerNewsList/index.jsx @@ -9,20 +9,25 @@ import './HackerNewsList.scss'; const propTypes = { feeds: PropTypes.object, // eslint-disable-line react/forbid-prop-types + isFetching: PropTypes.bool, }; const defaultProps = { feeds: Immutable.List(), + isFetching: false, }; -function HackerNewsList({ feeds }) { +function HackerNewsList({ feeds, isFetching }) { + const haveNoItems = !isFetching && feeds.size === 0; + return (
    - {feeds.map((feed, index) => ( + {haveNoItems &&

    There are no items to show.

    } + {!isFetching && feeds.map((feed, index) => (
  • {index + 1} diff --git a/app/components/Pagination/Pagination.spec.jsx b/app/components/Pagination/Pagination.spec.jsx index a71cce2..6774da5 100644 --- a/app/components/Pagination/Pagination.spec.jsx +++ b/app/components/Pagination/Pagination.spec.jsx @@ -9,6 +9,18 @@ describe('', () => { expect(wrapper).toMatchSnapshot(); }); + it('should have .Pagination__button--disabled when props.currentPage === 1', () => { + const wrapper = shallow(); + const button = wrapper.find('.Pagination__button--disabled'); + expect(button.props()['data-button']).toBe('prev'); + }); + + it('should have .Pagination__button--disabled when props.currentPage === 10', () => { + const wrapper = shallow(); + const button = wrapper.find('.Pagination__button--disabled'); + expect(button.props()['data-button']).toBe('next'); + }); + it('should calls onPaginate with currentPage - 1 when simulate prev button click event', () => { const currentPage = 3; const onPaginate = jest.fn(); @@ -20,8 +32,8 @@ describe('', () => { /> )); wrapper.find('.Pagination__button').at(0).simulate('click'); - expect(onPaginate.mock.calls.length).toBe(1); - expect(onPaginate.mock.calls[0]).toEqual([currentPage - 1]); + expect(onPaginate).toHaveBeenCalledTimes(1); + expect(onPaginate).toHaveBeenCalledWith(currentPage - 1); }); it('should calls onPaginate with currentPage + 1 when simulate next button click event', () => { @@ -35,7 +47,7 @@ describe('', () => { /> )); wrapper.find('.Pagination__button').at(1).simulate('click'); - expect(onPaginate.mock.calls.length).toBe(1); - expect(onPaginate.mock.calls[0]).toEqual([currentPage + 1]); + expect(onPaginate).toHaveBeenCalledTimes(1); + expect(onPaginate).toHaveBeenCalledWith(currentPage + 1); }); }); diff --git a/app/components/Pagination/__snapshots__/Pagination.spec.jsx.snap b/app/components/Pagination/__snapshots__/Pagination.spec.jsx.snap index f439d37..b0d4e2d 100644 --- a/app/components/Pagination/__snapshots__/Pagination.spec.jsx.snap +++ b/app/components/Pagination/__snapshots__/Pagination.spec.jsx.snap @@ -9,6 +9,7 @@ exports[` should match snapshot when render default 1`] = ` > + 1 + / 10 + {currentPage} / 10 {' '}