mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-11 02:37:22 +03:00
Added progress bar to visits page when loading a lot of visits
This commit is contained in:
parent
9bdbe90716
commit
07d3567244
9 changed files with 59 additions and 14 deletions
|
@ -61,3 +61,7 @@ body,
|
||||||
background-color: darken($mainColor, 12%);
|
background-color: darken($mainColor, 12%);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.progress-bar {
|
||||||
|
background-color: $mainColor;
|
||||||
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { isEmpty, propEq, values } from 'ramda';
|
import { isEmpty, propEq, values } from 'ramda';
|
||||||
import React, { useState, useEffect, useMemo } from 'react';
|
import React, { useState, useEffect, useMemo } from 'react';
|
||||||
import { Button, Card, Collapse } from 'reactstrap';
|
import { Button, Card, Collapse, Progress } from 'reactstrap';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
|
@ -59,7 +59,7 @@ const VisitsStats = ({ processStatsFromVisits, normalizeVisits }, OpenMapModalBt
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const { visits, loading, loadingLarge, error } = visitsInfo;
|
const { visits, loading, loadingLarge, error, progress } = visitsInfo;
|
||||||
const showTableControls = !loading && visits.length > 0;
|
const showTableControls = !loading && visits.length > 0;
|
||||||
const normalizedVisits = useMemo(() => normalizeVisits(visits), [ visits ]);
|
const normalizedVisits = useMemo(() => normalizeVisits(visits), [ visits ]);
|
||||||
const { os, browsers, referrers, countries, cities, citiesForMap } = useMemo(
|
const { os, browsers, referrers, countries, cities, citiesForMap } = useMemo(
|
||||||
|
@ -82,10 +82,17 @@ const VisitsStats = ({ processStatsFromVisits, normalizeVisits }, OpenMapModalBt
|
||||||
}, [ startDate, endDate ]);
|
}, [ startDate, endDate ]);
|
||||||
|
|
||||||
const renderVisitsContent = () => {
|
const renderVisitsContent = () => {
|
||||||
if (loading) {
|
if (loadingLarge) {
|
||||||
const message = loadingLarge ? 'This is going to take a while... :S' : 'Loading...';
|
return (
|
||||||
|
<Message loading>
|
||||||
|
This is going to take a while... :S
|
||||||
|
<Progress value={progress} striped={progress === 100} className="mt-3" />
|
||||||
|
</Message>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return <Message loading>{message}</Message>;
|
if (loading) {
|
||||||
|
return <Message loading />;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
|
|
|
@ -1,7 +1,11 @@
|
||||||
import { flatten, prop, range, splitEvery } from 'ramda';
|
import { flatten, prop, range, splitEvery } from 'ramda';
|
||||||
|
|
||||||
const ITEMS_PER_PAGE = 5000;
|
const ITEMS_PER_PAGE = 5000;
|
||||||
|
const PARALLEL_REQUESTS_COUNT = 4;
|
||||||
|
const PARALLEL_STARTING_PAGE = 2;
|
||||||
|
|
||||||
const isLastPage = ({ currentPage, pagesCount }) => currentPage >= pagesCount;
|
const isLastPage = ({ currentPage, pagesCount }) => currentPage >= pagesCount;
|
||||||
|
const calcProgress = (total, current) => current * 100 / total;
|
||||||
|
|
||||||
export const getVisitsWithLoader = async (visitsLoader, extraFinishActionData, actionMap, dispatch, getState) => {
|
export const getVisitsWithLoader = async (visitsLoader, extraFinishActionData, actionMap, dispatch, getState) => {
|
||||||
dispatch({ type: actionMap.start });
|
dispatch({ type: actionMap.start });
|
||||||
|
@ -15,12 +19,10 @@ export const getVisitsWithLoader = async (visitsLoader, extraFinishActionData, a
|
||||||
}
|
}
|
||||||
|
|
||||||
// If there are more pages, make requests in blocks of 4
|
// If there are more pages, make requests in blocks of 4
|
||||||
const parallelRequestsCount = 4;
|
const pagesRange = range(PARALLEL_STARTING_PAGE, pagination.pagesCount + 1);
|
||||||
const parallelStartingPage = 2;
|
const pagesBlocks = splitEvery(PARALLEL_REQUESTS_COUNT, pagesRange);
|
||||||
const pagesRange = range(parallelStartingPage, pagination.pagesCount + 1);
|
|
||||||
const pagesBlocks = splitEvery(parallelRequestsCount, pagesRange);
|
|
||||||
|
|
||||||
if (pagination.pagesCount - 1 > parallelRequestsCount) {
|
if (pagination.pagesCount - 1 > PARALLEL_REQUESTS_COUNT) {
|
||||||
dispatch({ type: actionMap.large });
|
dispatch({ type: actionMap.large });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,6 +38,8 @@ export const getVisitsWithLoader = async (visitsLoader, extraFinishActionData, a
|
||||||
|
|
||||||
const data = await loadVisitsInParallel(pagesBlocks[index]);
|
const data = await loadVisitsInParallel(pagesBlocks[index]);
|
||||||
|
|
||||||
|
dispatch({ type: actionMap.progress, progress: calcProgress(pagesBlocks.length, index + PARALLEL_STARTING_PAGE) });
|
||||||
|
|
||||||
if (index < pagesBlocks.length - 1) {
|
if (index < pagesBlocks.length - 1) {
|
||||||
return data.concat(await loadPagesBlocks(pagesBlocks, index + 1));
|
return data.concat(await loadPagesBlocks(pagesBlocks, index + 1));
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ export const GET_SHORT_URL_VISITS_ERROR = 'shlink/shortUrlVisits/GET_SHORT_URL_V
|
||||||
export const GET_SHORT_URL_VISITS = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS';
|
export const GET_SHORT_URL_VISITS = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS';
|
||||||
export const GET_SHORT_URL_VISITS_LARGE = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_LARGE';
|
export const GET_SHORT_URL_VISITS_LARGE = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_LARGE';
|
||||||
export const GET_SHORT_URL_VISITS_CANCEL = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_CANCEL';
|
export const GET_SHORT_URL_VISITS_CANCEL = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_CANCEL';
|
||||||
|
export const GET_SHORT_URL_VISITS_PROGRESS_CHANGED = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_PROGRESS_CHANGED';
|
||||||
/* eslint-enable padding-line-between-statements */
|
/* eslint-enable padding-line-between-statements */
|
||||||
|
|
||||||
export const shortUrlVisitsType = PropTypes.shape({ // TODO Should extend from VisitInfoType
|
export const shortUrlVisitsType = PropTypes.shape({ // TODO Should extend from VisitInfoType
|
||||||
|
@ -20,6 +21,7 @@ export const shortUrlVisitsType = PropTypes.shape({ // TODO Should extend from V
|
||||||
loading: PropTypes.bool,
|
loading: PropTypes.bool,
|
||||||
loadingLarge: PropTypes.bool,
|
loadingLarge: PropTypes.bool,
|
||||||
error: PropTypes.bool,
|
error: PropTypes.bool,
|
||||||
|
progress: PropTypes.number,
|
||||||
});
|
});
|
||||||
|
|
||||||
const initialState = {
|
const initialState = {
|
||||||
|
@ -30,6 +32,7 @@ const initialState = {
|
||||||
loadingLarge: false,
|
loadingLarge: false,
|
||||||
error: false,
|
error: false,
|
||||||
cancelLoad: false,
|
cancelLoad: false,
|
||||||
|
progress: 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default handleActions({
|
export default handleActions({
|
||||||
|
@ -43,6 +46,7 @@ export default handleActions({
|
||||||
}),
|
}),
|
||||||
[GET_SHORT_URL_VISITS_LARGE]: (state) => ({ ...state, loadingLarge: true }),
|
[GET_SHORT_URL_VISITS_LARGE]: (state) => ({ ...state, loadingLarge: true }),
|
||||||
[GET_SHORT_URL_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_SHORT_URL_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
|
[GET_SHORT_URL_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[CREATE_VISIT]: (state, { shortUrl, visit }) => { // eslint-disable-line object-shorthand
|
[CREATE_VISIT]: (state, { shortUrl, visit }) => { // eslint-disable-line object-shorthand
|
||||||
const { shortCode, domain, visits } = state;
|
const { shortCode, domain, visits } = state;
|
||||||
|
|
||||||
|
@ -63,6 +67,7 @@ export const getShortUrlVisits = (buildShlinkApiClient) => (shortCode, query = {
|
||||||
large: GET_SHORT_URL_VISITS_LARGE,
|
large: GET_SHORT_URL_VISITS_LARGE,
|
||||||
finish: GET_SHORT_URL_VISITS,
|
finish: GET_SHORT_URL_VISITS,
|
||||||
error: GET_SHORT_URL_VISITS_ERROR,
|
error: GET_SHORT_URL_VISITS_ERROR,
|
||||||
|
progress: GET_SHORT_URL_VISITS_PROGRESS_CHANGED,
|
||||||
};
|
};
|
||||||
|
|
||||||
return getVisitsWithLoader(visitsLoader, extraFinishActionData, actionMap, dispatch, getState);
|
return getVisitsWithLoader(visitsLoader, extraFinishActionData, actionMap, dispatch, getState);
|
||||||
|
|
|
@ -10,6 +10,7 @@ export const GET_TAG_VISITS_ERROR = 'shlink/tagVisits/GET_TAG_VISITS_ERROR';
|
||||||
export const GET_TAG_VISITS = 'shlink/tagVisits/GET_TAG_VISITS';
|
export const GET_TAG_VISITS = 'shlink/tagVisits/GET_TAG_VISITS';
|
||||||
export const GET_TAG_VISITS_LARGE = 'shlink/tagVisits/GET_TAG_VISITS_LARGE';
|
export const GET_TAG_VISITS_LARGE = 'shlink/tagVisits/GET_TAG_VISITS_LARGE';
|
||||||
export const GET_TAG_VISITS_CANCEL = 'shlink/tagVisits/GET_TAG_VISITS_CANCEL';
|
export const GET_TAG_VISITS_CANCEL = 'shlink/tagVisits/GET_TAG_VISITS_CANCEL';
|
||||||
|
export const GET_TAG_VISITS_PROGRESS_CHANGED = 'shlink/tagVisits/GET_TAG_VISITS_PROGRESS_CHANGED';
|
||||||
/* eslint-enable padding-line-between-statements */
|
/* eslint-enable padding-line-between-statements */
|
||||||
|
|
||||||
export const TagVisitsType = PropTypes.shape({ // TODO Should extend from VisitInfoType
|
export const TagVisitsType = PropTypes.shape({ // TODO Should extend from VisitInfoType
|
||||||
|
@ -18,6 +19,7 @@ export const TagVisitsType = PropTypes.shape({ // TODO Should extend from VisitI
|
||||||
loading: PropTypes.bool,
|
loading: PropTypes.bool,
|
||||||
loadingLarge: PropTypes.bool,
|
loadingLarge: PropTypes.bool,
|
||||||
error: PropTypes.bool,
|
error: PropTypes.bool,
|
||||||
|
progress: PropTypes.number,
|
||||||
});
|
});
|
||||||
|
|
||||||
const initialState = {
|
const initialState = {
|
||||||
|
@ -27,6 +29,7 @@ const initialState = {
|
||||||
loadingLarge: false,
|
loadingLarge: false,
|
||||||
error: false,
|
error: false,
|
||||||
cancelLoad: false,
|
cancelLoad: false,
|
||||||
|
progress: 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default handleActions({
|
export default handleActions({
|
||||||
|
@ -35,6 +38,7 @@ export default handleActions({
|
||||||
[GET_TAG_VISITS]: (state, { visits, tag }) => ({ ...initialState, visits, tag }),
|
[GET_TAG_VISITS]: (state, { visits, tag }) => ({ ...initialState, visits, tag }),
|
||||||
[GET_TAG_VISITS_LARGE]: (state) => ({ ...state, loadingLarge: true }),
|
[GET_TAG_VISITS_LARGE]: (state) => ({ ...state, loadingLarge: true }),
|
||||||
[GET_TAG_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
[GET_TAG_VISITS_CANCEL]: (state) => ({ ...state, cancelLoad: true }),
|
||||||
|
[GET_TAG_VISITS_PROGRESS_CHANGED]: (state, { progress }) => ({ ...state, progress }),
|
||||||
[CREATE_VISIT]: (state, { shortUrl, visit }) => { // eslint-disable-line object-shorthand
|
[CREATE_VISIT]: (state, { shortUrl, visit }) => { // eslint-disable-line object-shorthand
|
||||||
const { tag, visits } = state;
|
const { tag, visits } = state;
|
||||||
|
|
||||||
|
@ -55,6 +59,7 @@ export const getTagVisits = (buildShlinkApiClient) => (tag, query = {}) => (disp
|
||||||
large: GET_TAG_VISITS_LARGE,
|
large: GET_TAG_VISITS_LARGE,
|
||||||
finish: GET_TAG_VISITS,
|
finish: GET_TAG_VISITS,
|
||||||
error: GET_TAG_VISITS_ERROR,
|
error: GET_TAG_VISITS_ERROR,
|
||||||
|
progress: GET_TAG_VISITS_PROGRESS_CHANGED,
|
||||||
};
|
};
|
||||||
|
|
||||||
return getVisitsWithLoader(visitsLoader, extraFinishActionData, actionMap, dispatch, getState);
|
return getVisitsWithLoader(visitsLoader, extraFinishActionData, actionMap, dispatch, getState);
|
||||||
|
|
|
@ -21,4 +21,5 @@ export const VisitsInfoType = PropTypes.shape({
|
||||||
loading: PropTypes.bool,
|
loading: PropTypes.bool,
|
||||||
loadingLarge: PropTypes.bool,
|
loadingLarge: PropTypes.bool,
|
||||||
error: PropTypes.bool,
|
error: PropTypes.bool,
|
||||||
|
progress: PropTypes.number,
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow } from 'enzyme';
|
||||||
import { identity } from 'ramda';
|
import { identity } from 'ramda';
|
||||||
import { Card } from 'reactstrap';
|
import { Card, Progress } from 'reactstrap';
|
||||||
import createVisitStats from '../../src/visits/VisitsStats';
|
import createVisitStats from '../../src/visits/VisitsStats';
|
||||||
import Message from '../../src/utils/Message';
|
import Message from '../../src/utils/Message';
|
||||||
import GraphCard from '../../src/visits/GraphCard';
|
import GraphCard from '../../src/visits/GraphCard';
|
||||||
|
@ -35,17 +35,22 @@ describe('<VisitStats />', () => {
|
||||||
it('renders a preloader when visits are loading', () => {
|
it('renders a preloader when visits are loading', () => {
|
||||||
const wrapper = createComponent({ loading: true, visits: [] });
|
const wrapper = createComponent({ loading: true, visits: [] });
|
||||||
const loadingMessage = wrapper.find(Message);
|
const loadingMessage = wrapper.find(Message);
|
||||||
|
const progress = wrapper.find(Progress);
|
||||||
|
|
||||||
expect(loadingMessage).toHaveLength(1);
|
expect(loadingMessage).toHaveLength(1);
|
||||||
expect(loadingMessage.html()).toContain('Loading...');
|
expect(loadingMessage.html()).toContain('Loading...');
|
||||||
|
expect(progress).toHaveLength(0);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('renders a warning when loading large amounts of visits', () => {
|
it('renders a warning and progress bar when loading large amounts of visits', () => {
|
||||||
const wrapper = createComponent({ loading: true, loadingLarge: true, visits: [] });
|
const wrapper = createComponent({ loading: true, loadingLarge: true, visits: [], progress: 25 });
|
||||||
const loadingMessage = wrapper.find(Message);
|
const loadingMessage = wrapper.find(Message);
|
||||||
|
const progress = wrapper.find(Progress);
|
||||||
|
|
||||||
expect(loadingMessage).toHaveLength(1);
|
expect(loadingMessage).toHaveLength(1);
|
||||||
expect(loadingMessage.html()).toContain('This is going to take a while... :S');
|
expect(loadingMessage.html()).toContain('This is going to take a while... :S');
|
||||||
|
expect(progress).toHaveLength(1);
|
||||||
|
expect(progress.prop('value')).toEqual(25);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('renders an error message when visits could not be loaded', () => {
|
it('renders an error message when visits could not be loaded', () => {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import reducer, {
|
||||||
GET_SHORT_URL_VISITS,
|
GET_SHORT_URL_VISITS,
|
||||||
GET_SHORT_URL_VISITS_LARGE,
|
GET_SHORT_URL_VISITS_LARGE,
|
||||||
GET_SHORT_URL_VISITS_CANCEL,
|
GET_SHORT_URL_VISITS_CANCEL,
|
||||||
|
GET_SHORT_URL_VISITS_PROGRESS_CHANGED,
|
||||||
} from '../../../src/visits/reducers/shortUrlVisits';
|
} from '../../../src/visits/reducers/shortUrlVisits';
|
||||||
import { CREATE_VISIT } from '../../../src/visits/reducers/visitCreation';
|
import { CREATE_VISIT } from '../../../src/visits/reducers/visitCreation';
|
||||||
|
|
||||||
|
@ -66,6 +67,12 @@ describe('shortUrlVisitsReducer', () => {
|
||||||
|
|
||||||
expect(visits).toEqual(expectedVisits);
|
expect(visits).toEqual(expectedVisits);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('returns new progress on GET_SHORT_URL_VISITS_PROGRESS_CHANGED', () => {
|
||||||
|
const state = reducer({}, { type: GET_SHORT_URL_VISITS_PROGRESS_CHANGED, progress: 85 });
|
||||||
|
|
||||||
|
expect(state).toEqual({ progress: 85 });
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getShortUrlVisits', () => {
|
describe('getShortUrlVisits', () => {
|
||||||
|
@ -127,7 +134,7 @@ describe('shortUrlVisitsReducer', () => {
|
||||||
await getShortUrlVisits(() => ShlinkApiClient)('abc123')(dispatchMock, getState);
|
await getShortUrlVisits(() => ShlinkApiClient)('abc123')(dispatchMock, getState);
|
||||||
|
|
||||||
expect(ShlinkApiClient.getShortUrlVisits).toHaveBeenCalledTimes(expectedRequests);
|
expect(ShlinkApiClient.getShortUrlVisits).toHaveBeenCalledTimes(expectedRequests);
|
||||||
expect(dispatchMock).toHaveBeenNthCalledWith(2, expect.objectContaining({
|
expect(dispatchMock).toHaveBeenNthCalledWith(3, expect.objectContaining({
|
||||||
visits: [{}, {}, {}, {}, {}, {}],
|
visits: [{}, {}, {}, {}, {}, {}],
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
|
@ -6,6 +6,7 @@ import reducer, {
|
||||||
GET_TAG_VISITS,
|
GET_TAG_VISITS,
|
||||||
GET_TAG_VISITS_LARGE,
|
GET_TAG_VISITS_LARGE,
|
||||||
GET_TAG_VISITS_CANCEL,
|
GET_TAG_VISITS_CANCEL,
|
||||||
|
GET_TAG_VISITS_PROGRESS_CHANGED,
|
||||||
} from '../../../src/visits/reducers/tagVisits';
|
} from '../../../src/visits/reducers/tagVisits';
|
||||||
import { CREATE_VISIT } from '../../../src/visits/reducers/visitCreation';
|
import { CREATE_VISIT } from '../../../src/visits/reducers/visitCreation';
|
||||||
|
|
||||||
|
@ -66,6 +67,12 @@ describe('tagVisitsReducer', () => {
|
||||||
|
|
||||||
expect(visits).toEqual(expectedVisits);
|
expect(visits).toEqual(expectedVisits);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('returns new progress on GET_TAG_VISITS_PROGRESS_CHANGED', () => {
|
||||||
|
const state = reducer({}, { type: GET_TAG_VISITS_PROGRESS_CHANGED, progress: 85 });
|
||||||
|
|
||||||
|
expect(state).toEqual({ progress: 85 });
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('getTagVisits', () => {
|
describe('getTagVisits', () => {
|
||||||
|
|
Loading…
Reference in a new issue