mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-25 01:03:45 +03:00
Finished TS migration
This commit is contained in:
parent
73b854037d
commit
d4094e66b3
17 changed files with 417 additions and 414 deletions
|
@ -1,4 +1,4 @@
|
||||||
import React, { FC, useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import qs from 'qs';
|
import qs from 'qs';
|
||||||
import { RouteComponentProps } from 'react-router';
|
import { RouteComponentProps } from 'react-router';
|
||||||
import { MercureBoundProps, useMercureTopicBinding } from '../mercure/helpers';
|
import { MercureBoundProps, useMercureTopicBinding } from '../mercure/helpers';
|
||||||
|
@ -6,16 +6,17 @@ import { ShlinkVisitsParams } from '../utils/services/types';
|
||||||
import { ShortUrlVisits as ShortUrlVisitsState } from './reducers/shortUrlVisits';
|
import { ShortUrlVisits as ShortUrlVisitsState } from './reducers/shortUrlVisits';
|
||||||
import ShortUrlVisitsHeader from './ShortUrlVisitsHeader';
|
import ShortUrlVisitsHeader from './ShortUrlVisitsHeader';
|
||||||
import { ShortUrlDetail } from './reducers/shortUrlDetail';
|
import { ShortUrlDetail } from './reducers/shortUrlDetail';
|
||||||
|
import VisitsStats from './VisitsStats';
|
||||||
|
|
||||||
export interface ShortUrlVisitsProps extends RouteComponentProps<{ shortCode: string }>, MercureBoundProps {
|
export interface ShortUrlVisitsProps extends RouteComponentProps<{ shortCode: string }>, MercureBoundProps {
|
||||||
getShortUrlVisits: (shortCode: string, query?: ShlinkVisitsParams) => void;
|
getShortUrlVisits: (shortCode: string, query?: ShlinkVisitsParams) => void;
|
||||||
shortUrlVisits: ShortUrlVisitsState;
|
shortUrlVisits: ShortUrlVisitsState;
|
||||||
getShortUrlDetail: Function;
|
getShortUrlDetail: Function;
|
||||||
shortUrlDetail: ShortUrlDetail;
|
shortUrlDetail: ShortUrlDetail;
|
||||||
cancelGetShortUrlVisits: Function;
|
cancelGetShortUrlVisits: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ShortUrlVisits = (VisitsStats: FC<any>) => ({ // TODO Use VisitsStatsProps once available
|
const ShortUrlVisits = ({
|
||||||
history: { goBack },
|
history: { goBack },
|
||||||
match,
|
match,
|
||||||
location: { search },
|
location: { search },
|
||||||
|
@ -32,7 +33,7 @@ const ShortUrlVisits = (VisitsStats: FC<any>) => ({ // TODO Use VisitsStatsProps
|
||||||
const { shortCode } = params;
|
const { shortCode } = params;
|
||||||
const { domain } = qs.parse(search, { ignoreQueryPrefix: true }) as { domain?: string };
|
const { domain } = qs.parse(search, { ignoreQueryPrefix: true }) as { domain?: string };
|
||||||
|
|
||||||
const loadVisits = (dates: Partial<ShlinkVisitsParams>) => getShortUrlVisits(shortCode, { ...dates, domain });
|
const loadVisits = (params: Partial<ShlinkVisitsParams>) => getShortUrlVisits(shortCode, { ...params, domain });
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
getShortUrlDetail(shortCode, domain);
|
getShortUrlDetail(shortCode, domain);
|
||||||
|
|
|
@ -1,17 +1,19 @@
|
||||||
import React, { FC } from 'react';
|
import React from 'react';
|
||||||
import { RouteComponentProps } from 'react-router';
|
import { RouteComponentProps } from 'react-router';
|
||||||
import { MercureBoundProps, useMercureTopicBinding } from '../mercure/helpers';
|
import { MercureBoundProps, useMercureTopicBinding } from '../mercure/helpers';
|
||||||
import ColorGenerator from '../utils/services/ColorGenerator';
|
import ColorGenerator from '../utils/services/ColorGenerator';
|
||||||
|
import { ShlinkVisitsParams } from '../utils/services/types';
|
||||||
import { TagVisits as TagVisitsState } from './reducers/tagVisits';
|
import { TagVisits as TagVisitsState } from './reducers/tagVisits';
|
||||||
import TagVisitsHeader from './TagVisitsHeader';
|
import TagVisitsHeader from './TagVisitsHeader';
|
||||||
|
import VisitsStats from './VisitsStats';
|
||||||
|
|
||||||
export interface TagVisitsProps extends RouteComponentProps<{ tag: string }>, MercureBoundProps {
|
export interface TagVisitsProps extends RouteComponentProps<{ tag: string }>, MercureBoundProps {
|
||||||
getTagVisits: (tag: string, query: any) => void;
|
getTagVisits: (tag: string, query: any) => void;
|
||||||
tagVisits: TagVisitsState;
|
tagVisits: TagVisitsState;
|
||||||
cancelGetTagVisits: Function;
|
cancelGetTagVisits: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const TagVisits = (VisitsStats: FC<any>, colorGenerator: ColorGenerator) => ({ // TODO Use VisitsStatsProps once available
|
const TagVisits = (colorGenerator: ColorGenerator) => ({
|
||||||
history: { goBack },
|
history: { goBack },
|
||||||
match,
|
match,
|
||||||
getTagVisits,
|
getTagVisits,
|
||||||
|
@ -23,7 +25,7 @@ const TagVisits = (VisitsStats: FC<any>, colorGenerator: ColorGenerator) => ({ /
|
||||||
}: TagVisitsProps) => {
|
}: TagVisitsProps) => {
|
||||||
const { params } = match;
|
const { params } = match;
|
||||||
const { tag } = params;
|
const { tag } = params;
|
||||||
const loadVisits = (dates: any) => getTagVisits(tag, dates);
|
const loadVisits = (params: ShlinkVisitsParams) => getTagVisits(tag, params);
|
||||||
|
|
||||||
useMercureTopicBinding(mercureInfo, 'https://shlink.io/new-visit', createNewVisit, loadMercureInfo);
|
useMercureTopicBinding(mercureInfo, 'https://shlink.io/new-visit', createNewVisit, loadMercureInfo);
|
||||||
|
|
||||||
|
|
|
@ -1,247 +0,0 @@
|
||||||
import { isEmpty, propEq, values } from 'ramda';
|
|
||||||
import React, { useState, useEffect, useMemo } from 'react';
|
|
||||||
import { Button, Card, Collapse, Progress } from 'reactstrap';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import classNames from 'classnames';
|
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
|
||||||
import { faChevronDown as chevronDown } from '@fortawesome/free-solid-svg-icons';
|
|
||||||
import DateRangeRow from '../utils/DateRangeRow';
|
|
||||||
import Message from '../utils/Message';
|
|
||||||
import { formatDate } from '../utils/helpers/date';
|
|
||||||
import { useToggle } from '../utils/helpers/hooks';
|
|
||||||
import SortableBarGraph from './helpers/SortableBarGraph';
|
|
||||||
import GraphCard from './helpers/GraphCard';
|
|
||||||
import LineChartCard from './helpers/LineChartCard';
|
|
||||||
import VisitsTable from './VisitsTable';
|
|
||||||
import { VisitsInfoType } from './types';
|
|
||||||
import OpenMapModalBtn from './helpers/OpenMapModalBtn';
|
|
||||||
|
|
||||||
const propTypes = {
|
|
||||||
children: PropTypes.node,
|
|
||||||
getVisits: PropTypes.func,
|
|
||||||
visitsInfo: VisitsInfoType,
|
|
||||||
cancelGetVisits: PropTypes.func,
|
|
||||||
matchMedia: PropTypes.func,
|
|
||||||
};
|
|
||||||
|
|
||||||
const highlightedVisitsToStats = (highlightedVisits, prop) => highlightedVisits.reduce((acc, highlightedVisit) => {
|
|
||||||
if (!acc[highlightedVisit[prop]]) {
|
|
||||||
acc[highlightedVisit[prop]] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
acc[highlightedVisit[prop]] += 1;
|
|
||||||
|
|
||||||
return acc;
|
|
||||||
}, {});
|
|
||||||
const format = formatDate();
|
|
||||||
let selectedBar;
|
|
||||||
|
|
||||||
const VisitsStats = ({ processStatsFromVisits, normalizeVisits }) => {
|
|
||||||
const VisitsStatsComp = ({ children, visitsInfo, getVisits, cancelGetVisits, matchMedia = window.matchMedia }) => {
|
|
||||||
const [ startDate, setStartDate ] = useState(undefined);
|
|
||||||
const [ endDate, setEndDate ] = useState(undefined);
|
|
||||||
const [ showTable, toggleTable ] = useToggle();
|
|
||||||
const [ tableIsSticky, , setSticky, unsetSticky ] = useToggle();
|
|
||||||
const [ highlightedVisits, setHighlightedVisits ] = useState([]);
|
|
||||||
const [ highlightedLabel, setHighlightedLabel ] = useState();
|
|
||||||
const [ isMobileDevice, setIsMobileDevice ] = useState(false);
|
|
||||||
const determineIsMobileDevice = () => setIsMobileDevice(matchMedia('(max-width: 991px)').matches);
|
|
||||||
const setSelectedVisits = (selectedVisits) => {
|
|
||||||
selectedBar = undefined;
|
|
||||||
setHighlightedVisits(selectedVisits);
|
|
||||||
};
|
|
||||||
const highlightVisitsForProp = (prop) => (value) => {
|
|
||||||
const newSelectedBar = `${prop}_${value}`;
|
|
||||||
|
|
||||||
if (selectedBar === newSelectedBar) {
|
|
||||||
setHighlightedVisits([]);
|
|
||||||
setHighlightedLabel(undefined);
|
|
||||||
selectedBar = undefined;
|
|
||||||
} else {
|
|
||||||
setHighlightedVisits(normalizedVisits.filter(propEq(prop, value)));
|
|
||||||
setHighlightedLabel(value);
|
|
||||||
selectedBar = newSelectedBar;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const { visits, loading, loadingLarge, error, progress } = visitsInfo;
|
|
||||||
const showTableControls = !loading && visits.length > 0;
|
|
||||||
const normalizedVisits = useMemo(() => normalizeVisits(visits), [ visits ]);
|
|
||||||
const { os, browsers, referrers, countries, cities, citiesForMap } = useMemo(
|
|
||||||
() => processStatsFromVisits(normalizedVisits),
|
|
||||||
[ normalizedVisits ],
|
|
||||||
);
|
|
||||||
const mapLocations = values(citiesForMap);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
determineIsMobileDevice();
|
|
||||||
window.addEventListener('resize', determineIsMobileDevice);
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
cancelGetVisits();
|
|
||||||
window.removeEventListener('resize', determineIsMobileDevice);
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
useEffect(() => {
|
|
||||||
getVisits({ startDate: format(startDate), endDate: format(endDate) });
|
|
||||||
}, [ startDate, endDate ]);
|
|
||||||
|
|
||||||
const renderVisitsContent = () => {
|
|
||||||
if (loadingLarge) {
|
|
||||||
return (
|
|
||||||
<Message loading>
|
|
||||||
This is going to take a while... :S
|
|
||||||
<Progress value={progress} striped={progress === 100} className="mt-3" />
|
|
||||||
</Message>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (loading) {
|
|
||||||
return <Message loading />;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
return (
|
|
||||||
<Card className="mt-4" body inverse color="danger">
|
|
||||||
An error occurred while loading visits :(
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isEmpty(visits)) {
|
|
||||||
return <Message>There are no visits matching current filter :(</Message>;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="row">
|
|
||||||
<div className="col-12 mt-4">
|
|
||||||
<LineChartCard
|
|
||||||
title="Visits during time"
|
|
||||||
visits={visits}
|
|
||||||
highlightedVisits={highlightedVisits}
|
|
||||||
highlightedLabel={highlightedLabel}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="col-xl-4 col-lg-6 mt-4">
|
|
||||||
<GraphCard title="Operating systems" stats={os} />
|
|
||||||
</div>
|
|
||||||
<div className="col-xl-4 col-lg-6 mt-4">
|
|
||||||
<GraphCard title="Browsers" stats={browsers} />
|
|
||||||
</div>
|
|
||||||
<div className="col-xl-4 mt-4">
|
|
||||||
<SortableBarGraph
|
|
||||||
title="Referrers"
|
|
||||||
stats={referrers}
|
|
||||||
withPagination={false}
|
|
||||||
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'referer')}
|
|
||||||
highlightedLabel={highlightedLabel}
|
|
||||||
sortingItems={{
|
|
||||||
name: 'Referrer name',
|
|
||||||
amount: 'Visits amount',
|
|
||||||
}}
|
|
||||||
onClick={highlightVisitsForProp('referer')}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="col-lg-6 mt-4">
|
|
||||||
<SortableBarGraph
|
|
||||||
title="Countries"
|
|
||||||
stats={countries}
|
|
||||||
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'country')}
|
|
||||||
highlightedLabel={highlightedLabel}
|
|
||||||
sortingItems={{
|
|
||||||
name: 'Country name',
|
|
||||||
amount: 'Visits amount',
|
|
||||||
}}
|
|
||||||
onClick={highlightVisitsForProp('country')}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="col-lg-6 mt-4">
|
|
||||||
<SortableBarGraph
|
|
||||||
title="Cities"
|
|
||||||
stats={cities}
|
|
||||||
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'city')}
|
|
||||||
highlightedLabel={highlightedLabel}
|
|
||||||
extraHeaderContent={(activeCities) =>
|
|
||||||
mapLocations.length > 0 &&
|
|
||||||
<OpenMapModalBtn modalTitle="Cities" locations={mapLocations} activeCities={activeCities} />
|
|
||||||
}
|
|
||||||
sortingItems={{
|
|
||||||
name: 'City name',
|
|
||||||
amount: 'Visits amount',
|
|
||||||
}}
|
|
||||||
onClick={highlightVisitsForProp('city')}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<React.Fragment>
|
|
||||||
{children}
|
|
||||||
|
|
||||||
<section className="mt-4">
|
|
||||||
<div className="row flex-md-row-reverse">
|
|
||||||
<div className="col-lg-7 col-xl-6">
|
|
||||||
<DateRangeRow
|
|
||||||
disabled={loading}
|
|
||||||
startDate={startDate}
|
|
||||||
endDate={endDate}
|
|
||||||
onStartDateChange={setStartDate}
|
|
||||||
onEndDateChange={setEndDate}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="col-lg-5 col-xl-6 mt-4 mt-lg-0">
|
|
||||||
{showTableControls && (
|
|
||||||
<span className={classNames({ row: isMobileDevice })}>
|
|
||||||
<span className={classNames({ 'col-6': isMobileDevice })}>
|
|
||||||
<Button outline color="primary" block={isMobileDevice} onClick={toggleTable}>
|
|
||||||
{showTable ? 'Hide' : 'Show'} table
|
|
||||||
<FontAwesomeIcon icon={chevronDown} rotation={showTable ? 180 : undefined} className="ml-2" />
|
|
||||||
</Button>
|
|
||||||
</span>
|
|
||||||
<span className={classNames({ 'col-6': isMobileDevice, 'ml-2': !isMobileDevice })}>
|
|
||||||
<Button
|
|
||||||
outline
|
|
||||||
disabled={highlightedVisits.length === 0}
|
|
||||||
block={isMobileDevice}
|
|
||||||
onClick={() => setSelectedVisits([])}
|
|
||||||
>
|
|
||||||
Reset selection
|
|
||||||
</Button>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{showTableControls && (
|
|
||||||
<Collapse
|
|
||||||
isOpen={showTable}
|
|
||||||
// Enable stickiness only when there's no CSS animation, to avoid weird rendering effects
|
|
||||||
onEntered={setSticky}
|
|
||||||
onExiting={unsetSticky}
|
|
||||||
>
|
|
||||||
<VisitsTable
|
|
||||||
visits={normalizedVisits}
|
|
||||||
selectedVisits={highlightedVisits}
|
|
||||||
setSelectedVisits={setSelectedVisits}
|
|
||||||
isSticky={tableIsSticky}
|
|
||||||
/>
|
|
||||||
</Collapse>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<section>
|
|
||||||
{renderVisitsContent()}
|
|
||||||
</section>
|
|
||||||
</React.Fragment>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
VisitsStatsComp.propTypes = propTypes;
|
|
||||||
|
|
||||||
return VisitsStatsComp;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default VisitsStats;
|
|
250
src/visits/VisitsStats.tsx
Normal file
250
src/visits/VisitsStats.tsx
Normal file
|
@ -0,0 +1,250 @@
|
||||||
|
import { isEmpty, propEq, values } from 'ramda';
|
||||||
|
import React, { useState, useEffect, useMemo, FC } from 'react';
|
||||||
|
import { Button, Card, Collapse, Progress } from 'reactstrap';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
|
import { faChevronDown as chevronDown } from '@fortawesome/free-solid-svg-icons';
|
||||||
|
import moment from 'moment';
|
||||||
|
import DateRangeRow from '../utils/DateRangeRow';
|
||||||
|
import Message from '../utils/Message';
|
||||||
|
import { formatDate } from '../utils/helpers/date';
|
||||||
|
import { useToggle } from '../utils/helpers/hooks';
|
||||||
|
import { ShlinkVisitsParams } from '../utils/services/types';
|
||||||
|
import SortableBarGraph from './helpers/SortableBarGraph';
|
||||||
|
import GraphCard from './helpers/GraphCard';
|
||||||
|
import LineChartCard from './helpers/LineChartCard';
|
||||||
|
import VisitsTable from './VisitsTable';
|
||||||
|
import { NormalizedVisit, Stats, VisitsInfo } from './types';
|
||||||
|
import OpenMapModalBtn from './helpers/OpenMapModalBtn';
|
||||||
|
import { normalizeVisits, processStatsFromVisits } from './services/VisitsParser';
|
||||||
|
|
||||||
|
export interface VisitsStatsProps {
|
||||||
|
matchMedia?: (query: string) => MediaQueryList;
|
||||||
|
getVisits: (params: Partial<ShlinkVisitsParams>) => void;
|
||||||
|
visitsInfo: VisitsInfo;
|
||||||
|
cancelGetVisits: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
type HighlightableProps = 'referer' | 'country' | 'city';
|
||||||
|
|
||||||
|
const highlightedVisitsToStats = (
|
||||||
|
highlightedVisits: NormalizedVisit[],
|
||||||
|
prop: HighlightableProps,
|
||||||
|
): Stats => highlightedVisits.reduce<Stats>((acc, highlightedVisit) => {
|
||||||
|
if (!acc[highlightedVisit[prop]]) {
|
||||||
|
acc[highlightedVisit[prop]] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
acc[highlightedVisit[prop]] += 1;
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
const format = formatDate();
|
||||||
|
let selectedBar: string | undefined;
|
||||||
|
|
||||||
|
const VisitsStats: FC<VisitsStatsProps> = (
|
||||||
|
{ children, visitsInfo, getVisits, cancelGetVisits, matchMedia = window.matchMedia },
|
||||||
|
) => {
|
||||||
|
const [ startDate, setStartDate ] = useState<moment.Moment | null>(null);
|
||||||
|
const [ endDate, setEndDate ] = useState<moment.Moment | null>(null);
|
||||||
|
const [ showTable, toggleTable ] = useToggle();
|
||||||
|
const [ tableIsSticky, , setSticky, unsetSticky ] = useToggle();
|
||||||
|
const [ highlightedVisits, setHighlightedVisits ] = useState<NormalizedVisit[]>([]);
|
||||||
|
const [ highlightedLabel, setHighlightedLabel ] = useState<string | undefined>();
|
||||||
|
const [ isMobileDevice, setIsMobileDevice ] = useState(false);
|
||||||
|
|
||||||
|
const { visits, loading, loadingLarge, error, progress } = visitsInfo;
|
||||||
|
const showTableControls = !loading && visits.length > 0;
|
||||||
|
const normalizedVisits = useMemo(() => normalizeVisits(visits), [ visits ]);
|
||||||
|
const { os, browsers, referrers, countries, cities, citiesForMap } = useMemo(
|
||||||
|
() => processStatsFromVisits(normalizedVisits),
|
||||||
|
[ normalizedVisits ],
|
||||||
|
);
|
||||||
|
const mapLocations = values(citiesForMap);
|
||||||
|
|
||||||
|
const determineIsMobileDevice = () => setIsMobileDevice(matchMedia('(max-width: 991px)').matches);
|
||||||
|
const setSelectedVisits = (selectedVisits: NormalizedVisit[]) => {
|
||||||
|
selectedBar = undefined;
|
||||||
|
setHighlightedVisits(selectedVisits);
|
||||||
|
};
|
||||||
|
const highlightVisitsForProp = (prop: HighlightableProps) => (value: string) => {
|
||||||
|
const newSelectedBar = `${prop}_${value}`;
|
||||||
|
|
||||||
|
if (selectedBar === newSelectedBar) {
|
||||||
|
setHighlightedVisits([]);
|
||||||
|
setHighlightedLabel(undefined);
|
||||||
|
selectedBar = undefined;
|
||||||
|
} else {
|
||||||
|
setHighlightedVisits(normalizedVisits.filter(propEq(prop, value)));
|
||||||
|
setHighlightedLabel(value);
|
||||||
|
selectedBar = newSelectedBar;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
determineIsMobileDevice();
|
||||||
|
window.addEventListener('resize', determineIsMobileDevice);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
cancelGetVisits();
|
||||||
|
window.removeEventListener('resize', determineIsMobileDevice);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
useEffect(() => {
|
||||||
|
getVisits({ startDate: format(startDate) ?? undefined, endDate: format(endDate) ?? undefined });
|
||||||
|
}, [ startDate, endDate ]);
|
||||||
|
|
||||||
|
const renderVisitsContent = () => {
|
||||||
|
if (loadingLarge) {
|
||||||
|
return (
|
||||||
|
<Message loading>
|
||||||
|
This is going to take a while... :S
|
||||||
|
<Progress value={progress} striped={progress === 100} className="mt-3" />
|
||||||
|
</Message>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <Message loading />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
return (
|
||||||
|
<Card className="mt-4" body inverse color="danger">
|
||||||
|
An error occurred while loading visits :(
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isEmpty(visits)) {
|
||||||
|
return <Message>There are no visits matching current filter :(</Message>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="row">
|
||||||
|
<div className="col-12 mt-4">
|
||||||
|
<LineChartCard
|
||||||
|
title="Visits during time"
|
||||||
|
visits={visits}
|
||||||
|
highlightedVisits={highlightedVisits}
|
||||||
|
highlightedLabel={highlightedLabel}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="col-xl-4 col-lg-6 mt-4">
|
||||||
|
<GraphCard title="Operating systems" stats={os} />
|
||||||
|
</div>
|
||||||
|
<div className="col-xl-4 col-lg-6 mt-4">
|
||||||
|
<GraphCard title="Browsers" stats={browsers} />
|
||||||
|
</div>
|
||||||
|
<div className="col-xl-4 mt-4">
|
||||||
|
<SortableBarGraph
|
||||||
|
title="Referrers"
|
||||||
|
stats={referrers}
|
||||||
|
withPagination={false}
|
||||||
|
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'referer')}
|
||||||
|
highlightedLabel={highlightedLabel}
|
||||||
|
sortingItems={{
|
||||||
|
name: 'Referrer name',
|
||||||
|
amount: 'Visits amount',
|
||||||
|
}}
|
||||||
|
onClick={highlightVisitsForProp('referer')}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="col-lg-6 mt-4">
|
||||||
|
<SortableBarGraph
|
||||||
|
title="Countries"
|
||||||
|
stats={countries}
|
||||||
|
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'country')}
|
||||||
|
highlightedLabel={highlightedLabel}
|
||||||
|
sortingItems={{
|
||||||
|
name: 'Country name',
|
||||||
|
amount: 'Visits amount',
|
||||||
|
}}
|
||||||
|
onClick={highlightVisitsForProp('country')}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="col-lg-6 mt-4">
|
||||||
|
<SortableBarGraph
|
||||||
|
title="Cities"
|
||||||
|
stats={cities}
|
||||||
|
highlightedStats={highlightedVisitsToStats(highlightedVisits, 'city')}
|
||||||
|
highlightedLabel={highlightedLabel}
|
||||||
|
extraHeaderContent={(activeCities: string[]) =>
|
||||||
|
mapLocations.length > 0 &&
|
||||||
|
<OpenMapModalBtn modalTitle="Cities" locations={mapLocations} activeCities={activeCities} />
|
||||||
|
}
|
||||||
|
sortingItems={{
|
||||||
|
name: 'City name',
|
||||||
|
amount: 'Visits amount',
|
||||||
|
}}
|
||||||
|
onClick={highlightVisitsForProp('city')}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<React.Fragment>
|
||||||
|
{children}
|
||||||
|
|
||||||
|
<section className="mt-4">
|
||||||
|
<div className="row flex-md-row-reverse">
|
||||||
|
<div className="col-lg-7 col-xl-6">
|
||||||
|
<DateRangeRow
|
||||||
|
disabled={loading}
|
||||||
|
startDate={startDate}
|
||||||
|
endDate={endDate}
|
||||||
|
onStartDateChange={setStartDate}
|
||||||
|
onEndDateChange={setEndDate}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="col-lg-5 col-xl-6 mt-4 mt-lg-0">
|
||||||
|
{showTableControls && (
|
||||||
|
<span className={classNames({ row: isMobileDevice })}>
|
||||||
|
<span className={classNames({ 'col-6': isMobileDevice })}>
|
||||||
|
<Button outline color="primary" block={isMobileDevice} onClick={toggleTable}>
|
||||||
|
{showTable ? 'Hide' : 'Show'} table
|
||||||
|
<FontAwesomeIcon icon={chevronDown} rotation={showTable ? 180 : undefined} className="ml-2" />
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
<span className={classNames({ 'col-6': isMobileDevice, 'ml-2': !isMobileDevice })}>
|
||||||
|
<Button
|
||||||
|
outline
|
||||||
|
disabled={highlightedVisits.length === 0}
|
||||||
|
block={isMobileDevice}
|
||||||
|
onClick={() => setSelectedVisits([])}
|
||||||
|
>
|
||||||
|
Reset selection
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
{showTableControls && (
|
||||||
|
<Collapse
|
||||||
|
isOpen={showTable}
|
||||||
|
// Enable stickiness only when there's no CSS animation, to avoid weird rendering effects
|
||||||
|
onEntered={setSticky}
|
||||||
|
onExiting={unsetSticky}
|
||||||
|
>
|
||||||
|
<VisitsTable
|
||||||
|
visits={normalizedVisits}
|
||||||
|
selectedVisits={highlightedVisits}
|
||||||
|
setSelectedVisits={setSelectedVisits}
|
||||||
|
isSticky={tableIsSticky}
|
||||||
|
/>
|
||||||
|
</Collapse>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<section>
|
||||||
|
{renderVisitsContent()}
|
||||||
|
</section>
|
||||||
|
</React.Fragment>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default VisitsStats;
|
|
@ -1,5 +1,4 @@
|
||||||
import React, { useEffect, useMemo, useState } from 'react';
|
import React, { useEffect, useMemo, useState } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import Moment from 'react-moment';
|
import Moment from 'react-moment';
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import { min, splitEvery } from 'ramda';
|
import { min, splitEvery } from 'ramda';
|
||||||
|
@ -11,35 +10,42 @@ import {
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
import SimplePaginator from '../common/SimplePaginator';
|
import SimplePaginator from '../common/SimplePaginator';
|
||||||
import SearchField from '../utils/SearchField';
|
import SearchField from '../utils/SearchField';
|
||||||
import { determineOrderDir } from '../utils/utils';
|
import { determineOrderDir, OrderDir } from '../utils/utils';
|
||||||
import { prettify } from '../utils/helpers/numbers';
|
import { prettify } from '../utils/helpers/numbers';
|
||||||
|
import { NormalizedVisit } from './types';
|
||||||
import './VisitsTable.scss';
|
import './VisitsTable.scss';
|
||||||
|
|
||||||
const NormalizedVisitType = PropTypes.shape({
|
interface VisitsTableProps {
|
||||||
|
visits: NormalizedVisit[];
|
||||||
|
selectedVisits?: NormalizedVisit[];
|
||||||
|
setSelectedVisits: (visits: NormalizedVisit[]) => void;
|
||||||
|
isSticky?: boolean;
|
||||||
|
matchMedia?: (query: string) => MediaQueryList;
|
||||||
|
}
|
||||||
|
|
||||||
});
|
type OrderableFields = 'date' | 'country' | 'city' | 'browser' | 'os' | 'referer';
|
||||||
|
|
||||||
const propTypes = {
|
interface Order {
|
||||||
visits: PropTypes.arrayOf(NormalizedVisitType).isRequired,
|
field?: OrderableFields;
|
||||||
selectedVisits: PropTypes.arrayOf(NormalizedVisitType),
|
dir?: OrderDir;
|
||||||
setSelectedVisits: PropTypes.func.isRequired,
|
}
|
||||||
isSticky: PropTypes.bool,
|
|
||||||
matchMedia: PropTypes.func,
|
|
||||||
};
|
|
||||||
|
|
||||||
const PAGE_SIZE = 20;
|
const PAGE_SIZE = 20;
|
||||||
const visitMatchesSearch = ({ browser, os, referer, country, city }, searchTerm) =>
|
const visitMatchesSearch = ({ browser, os, referer, country, city }: NormalizedVisit, searchTerm: string) =>
|
||||||
`${browser} ${os} ${referer} ${country} ${city}`.toLowerCase().includes(searchTerm.toLowerCase());
|
`${browser} ${os} ${referer} ${country} ${city}`.toLowerCase().includes(searchTerm.toLowerCase());
|
||||||
const searchVisits = (searchTerm, visits) => visits.filter((visit) => visitMatchesSearch(visit, searchTerm));
|
const searchVisits = (searchTerm: string, visits: NormalizedVisit[]) =>
|
||||||
const sortVisits = ({ field, dir }, visits) => visits.sort((a, b) => {
|
visits.filter((visit) => visitMatchesSearch(visit, searchTerm));
|
||||||
|
const sortVisits = ({ field, dir }: Order, visits: NormalizedVisit[]) => !field || !dir ? visits : visits.sort(
|
||||||
|
(a, b) => {
|
||||||
const greaterThan = dir === 'ASC' ? 1 : -1;
|
const greaterThan = dir === 'ASC' ? 1 : -1;
|
||||||
const smallerThan = dir === 'ASC' ? -1 : 1;
|
const smallerThan = dir === 'ASC' ? -1 : 1;
|
||||||
|
|
||||||
return a[field] > b[field] ? greaterThan : smallerThan;
|
return a[field] > b[field] ? greaterThan : smallerThan;
|
||||||
});
|
},
|
||||||
const calculateVisits = (allVisits, searchTerm, order) => {
|
);
|
||||||
|
const calculateVisits = (allVisits: NormalizedVisit[], searchTerm: string | undefined, order: Order) => {
|
||||||
const filteredVisits = searchTerm ? searchVisits(searchTerm, allVisits) : [ ...allVisits ];
|
const filteredVisits = searchTerm ? searchVisits(searchTerm, allVisits) : [ ...allVisits ];
|
||||||
const sortedVisits = order.dir ? sortVisits(order, filteredVisits) : filteredVisits;
|
const sortedVisits = sortVisits(order, filteredVisits);
|
||||||
const total = sortedVisits.length;
|
const total = sortedVisits.length;
|
||||||
const visitsGroups = splitEvery(PAGE_SIZE, sortedVisits);
|
const visitsGroups = splitEvery(PAGE_SIZE, sortedVisits);
|
||||||
|
|
||||||
|
@ -52,23 +58,24 @@ const VisitsTable = ({
|
||||||
setSelectedVisits,
|
setSelectedVisits,
|
||||||
isSticky = false,
|
isSticky = false,
|
||||||
matchMedia = window.matchMedia,
|
matchMedia = window.matchMedia,
|
||||||
}) => {
|
}: VisitsTableProps) => {
|
||||||
const headerCellsClass = classNames('visits-table__header-cell', {
|
const headerCellsClass = classNames('visits-table__header-cell', {
|
||||||
'visits-table__sticky': isSticky,
|
'visits-table__sticky': isSticky,
|
||||||
});
|
});
|
||||||
const matchMobile = () => matchMedia('(max-width: 767px)').matches;
|
const matchMobile = () => matchMedia('(max-width: 767px)').matches;
|
||||||
|
|
||||||
const [ isMobileDevice, setIsMobileDevice ] = useState(matchMobile());
|
const [ isMobileDevice, setIsMobileDevice ] = useState(matchMobile());
|
||||||
const [ searchTerm, setSearchTerm ] = useState(undefined);
|
const [ searchTerm, setSearchTerm ] = useState<string | undefined>(undefined);
|
||||||
const [ order, setOrder ] = useState({ field: undefined, dir: undefined });
|
const [ order, setOrder ] = useState<Order>({ field: undefined, dir: undefined });
|
||||||
const resultSet = useMemo(() => calculateVisits(visits, searchTerm, order), [ searchTerm, order ]);
|
const resultSet = useMemo(() => calculateVisits(visits, searchTerm, order), [ searchTerm, order ]);
|
||||||
|
|
||||||
const [ page, setPage ] = useState(1);
|
const [ page, setPage ] = useState(1);
|
||||||
const end = page * PAGE_SIZE;
|
const end = page * PAGE_SIZE;
|
||||||
const start = end - PAGE_SIZE;
|
const start = end - PAGE_SIZE;
|
||||||
|
|
||||||
const orderByColumn = (field) => () => setOrder({ field, dir: determineOrderDir(field, order.field, order.dir) });
|
const orderByColumn = (field: OrderableFields) =>
|
||||||
const renderOrderIcon = (field) => order.dir && order.field === field && (
|
() => setOrder({ field, dir: determineOrderDir(field, order.field, order.dir) });
|
||||||
|
const renderOrderIcon = (field: OrderableFields) => order.dir && order.field === field && (
|
||||||
<FontAwesomeIcon
|
<FontAwesomeIcon
|
||||||
icon={order.dir === 'ASC' ? caretUpIcon : caretDownIcon}
|
icon={order.dir === 'ASC' ? caretUpIcon : caretDownIcon}
|
||||||
className="visits-table__header-icon"
|
className="visits-table__header-icon"
|
||||||
|
@ -201,6 +208,4 @@ const VisitsTable = ({
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
VisitsTable.propTypes = propTypes;
|
|
||||||
|
|
||||||
export default VisitsTable;
|
export default VisitsTable;
|
|
@ -12,7 +12,7 @@ import { Line } from 'react-chartjs-2';
|
||||||
import { always, cond, reverse } from 'ramda';
|
import { always, cond, reverse } from 'ramda';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import { ChartData, ChartDataSets } from 'chart.js';
|
import { ChartData, ChartDataSets } from 'chart.js';
|
||||||
import { Stats, Visit } from '../types';
|
import { NormalizedVisit, Stats, Visit } from '../types';
|
||||||
import { fillTheGaps } from '../../utils/helpers/visits';
|
import { fillTheGaps } from '../../utils/helpers/visits';
|
||||||
import { useToggle } from '../../utils/helpers/hooks';
|
import { useToggle } from '../../utils/helpers/hooks';
|
||||||
import { rangeOf } from '../../utils/utils';
|
import { rangeOf } from '../../utils/utils';
|
||||||
|
@ -23,7 +23,7 @@ interface LineChartCardProps {
|
||||||
title: string;
|
title: string;
|
||||||
highlightedLabel?: string;
|
highlightedLabel?: string;
|
||||||
visits: Visit[];
|
visits: Visit[];
|
||||||
highlightedVisits: Visit[];
|
highlightedVisits: NormalizedVisit[];
|
||||||
}
|
}
|
||||||
|
|
||||||
type Step = 'monthly' | 'weekly' | 'daily' | 'hourly';
|
type Step = 'monthly' | 'weekly' | 'daily' | 'hourly';
|
||||||
|
@ -66,13 +66,16 @@ const determineInitialStep = (oldestVisitDate: string): Step => {
|
||||||
return matcher() ?? 'monthly';
|
return matcher() ?? 'monthly';
|
||||||
};
|
};
|
||||||
|
|
||||||
const groupVisitsByStep = (step: Step, visits: Visit[]): Stats => visits.reduce<Stats>((acc, visit) => {
|
const groupVisitsByStep = (step: Step, visits: (Visit | NormalizedVisit)[]): Stats => visits.reduce<Stats>(
|
||||||
|
(acc, visit) => {
|
||||||
const key = STEP_TO_DATE_FORMAT[step](visit.date);
|
const key = STEP_TO_DATE_FORMAT[step](visit.date);
|
||||||
|
|
||||||
acc[key] = acc[key] ? acc[key] + 1 : 1;
|
acc[key] = acc[key] ? acc[key] + 1 : 1;
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, {});
|
},
|
||||||
|
{},
|
||||||
|
);
|
||||||
|
|
||||||
const generateLabels = (step: Step, visits: Visit[]): string[] => {
|
const generateLabels = (step: Step, visits: Visit[]): string[] => {
|
||||||
const unit = STEP_TO_DATE_UNIT_MAP[step];
|
const unit = STEP_TO_DATE_UNIT_MAP[step];
|
||||||
|
|
|
@ -77,3 +77,8 @@ export const normalizeVisits = map(({ userAgent, date, referer, visitLocation }:
|
||||||
latitude: visitLocation?.latitude,
|
latitude: visitLocation?.latitude,
|
||||||
longitude: visitLocation?.longitude,
|
longitude: visitLocation?.longitude,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
export interface VisitsParser {
|
||||||
|
processStatsFromVisits: (normalizedVisits: NormalizedVisit[]) => VisitsStats;
|
||||||
|
normalizeVisits: (visits: Visit[]) => NormalizedVisit[];
|
||||||
|
}
|
||||||
|
|
|
@ -3,7 +3,6 @@ import ShortUrlVisits from '../ShortUrlVisits';
|
||||||
import { cancelGetShortUrlVisits, getShortUrlVisits } from '../reducers/shortUrlVisits';
|
import { cancelGetShortUrlVisits, getShortUrlVisits } from '../reducers/shortUrlVisits';
|
||||||
import { getShortUrlDetail } from '../reducers/shortUrlDetail';
|
import { getShortUrlDetail } from '../reducers/shortUrlDetail';
|
||||||
import MapModal from '../helpers/MapModal';
|
import MapModal from '../helpers/MapModal';
|
||||||
import VisitsStats from '../VisitsStats';
|
|
||||||
import { createNewVisit } from '../reducers/visitCreation';
|
import { createNewVisit } from '../reducers/visitCreation';
|
||||||
import { cancelGetTagVisits, getTagVisits } from '../reducers/tagVisits';
|
import { cancelGetTagVisits, getTagVisits } from '../reducers/tagVisits';
|
||||||
import TagVisits from '../TagVisits';
|
import TagVisits from '../TagVisits';
|
||||||
|
@ -13,13 +12,12 @@ import * as visitsParser from './VisitsParser';
|
||||||
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
const provideServices = (bottle: Bottle, connect: ConnectDecorator) => {
|
||||||
// Components
|
// Components
|
||||||
bottle.serviceFactory('MapModal', () => MapModal);
|
bottle.serviceFactory('MapModal', () => MapModal);
|
||||||
bottle.serviceFactory('VisitsStats', VisitsStats, 'VisitsParser');
|
bottle.serviceFactory('ShortUrlVisits', () => ShortUrlVisits);
|
||||||
bottle.serviceFactory('ShortUrlVisits', ShortUrlVisits, 'VisitsStats');
|
|
||||||
bottle.decorator('ShortUrlVisits', connect(
|
bottle.decorator('ShortUrlVisits', connect(
|
||||||
[ 'shortUrlVisits', 'shortUrlDetail', 'mercureInfo' ],
|
[ 'shortUrlVisits', 'shortUrlDetail', 'mercureInfo' ],
|
||||||
[ 'getShortUrlVisits', 'getShortUrlDetail', 'cancelGetShortUrlVisits', 'createNewVisit', 'loadMercureInfo' ],
|
[ 'getShortUrlVisits', 'getShortUrlDetail', 'cancelGetShortUrlVisits', 'createNewVisit', 'loadMercureInfo' ],
|
||||||
));
|
));
|
||||||
bottle.serviceFactory('TagVisits', TagVisits, 'VisitsStats', 'ColorGenerator');
|
bottle.serviceFactory('TagVisits', TagVisits, 'ColorGenerator');
|
||||||
bottle.decorator('TagVisits', connect(
|
bottle.decorator('TagVisits', connect(
|
||||||
[ 'tagVisits', 'mercureInfo' ],
|
[ 'tagVisits', 'mercureInfo' ],
|
||||||
[ 'getTagVisits', 'cancelGetTagVisits', 'createNewVisit', 'loadMercureInfo' ],
|
[ 'getTagVisits', 'cancelGetTagVisits', 'createNewVisit', 'loadMercureInfo' ],
|
||||||
|
|
|
@ -1,33 +1,6 @@
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { Action } from 'redux';
|
import { Action } from 'redux';
|
||||||
import { ShortUrl } from '../../short-urls/data';
|
import { ShortUrl } from '../../short-urls/data';
|
||||||
|
|
||||||
/** @deprecated Use Visit interface instead */
|
|
||||||
export const VisitType = PropTypes.shape({
|
|
||||||
referer: PropTypes.string,
|
|
||||||
date: PropTypes.string,
|
|
||||||
userAgent: PropTypes.string,
|
|
||||||
visitLocation: PropTypes.shape({
|
|
||||||
countryCode: PropTypes.string,
|
|
||||||
countryName: PropTypes.string,
|
|
||||||
regionName: PropTypes.string,
|
|
||||||
cityName: PropTypes.string,
|
|
||||||
latitude: PropTypes.number,
|
|
||||||
longitude: PropTypes.number,
|
|
||||||
timezone: PropTypes.string,
|
|
||||||
isEmpty: PropTypes.bool,
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
/** @deprecated Use VisitsInfo interface instead */
|
|
||||||
export const VisitsInfoType = PropTypes.shape({
|
|
||||||
visits: PropTypes.arrayOf(VisitType),
|
|
||||||
loading: PropTypes.bool,
|
|
||||||
loadingLarge: PropTypes.bool,
|
|
||||||
error: PropTypes.bool,
|
|
||||||
progress: PropTypes.number,
|
|
||||||
});
|
|
||||||
|
|
||||||
export interface VisitsInfo {
|
export interface VisitsInfo {
|
||||||
visits: Visit[];
|
visits: Visit[];
|
||||||
loading: boolean;
|
loading: boolean;
|
||||||
|
|
|
@ -4,10 +4,11 @@ import { identity } from 'ramda';
|
||||||
import { Mock } from 'ts-mockery';
|
import { Mock } from 'ts-mockery';
|
||||||
import { History, Location } from 'history';
|
import { History, Location } from 'history';
|
||||||
import { match } from 'react-router'; // eslint-disable-line @typescript-eslint/no-unused-vars
|
import { match } from 'react-router'; // eslint-disable-line @typescript-eslint/no-unused-vars
|
||||||
import createShortUrlVisits, { ShortUrlVisitsProps } from '../../src/visits/ShortUrlVisits';
|
import ShortUrlVisits, { ShortUrlVisitsProps } from '../../src/visits/ShortUrlVisits';
|
||||||
import ShortUrlVisitsHeader from '../../src/visits/ShortUrlVisitsHeader';
|
import ShortUrlVisitsHeader from '../../src/visits/ShortUrlVisitsHeader';
|
||||||
import { ShortUrlVisits as ShortUrlVisitsState } from '../../src/visits/reducers/shortUrlVisits';
|
import { ShortUrlVisits as ShortUrlVisitsState } from '../../src/visits/reducers/shortUrlVisits';
|
||||||
import { ShortUrlDetail } from '../../src/visits/reducers/shortUrlDetail';
|
import { ShortUrlDetail } from '../../src/visits/reducers/shortUrlDetail';
|
||||||
|
import VisitsStats from '../../src/visits/VisitsStats';
|
||||||
|
|
||||||
describe('<ShortUrlVisits />', () => {
|
describe('<ShortUrlVisits />', () => {
|
||||||
let wrapper: ShallowWrapper;
|
let wrapper: ShallowWrapper;
|
||||||
|
@ -19,11 +20,8 @@ describe('<ShortUrlVisits />', () => {
|
||||||
const history = Mock.of<History>({
|
const history = Mock.of<History>({
|
||||||
goBack: jest.fn(),
|
goBack: jest.fn(),
|
||||||
});
|
});
|
||||||
const VisitsStats = jest.fn();
|
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
const ShortUrlVisits = createShortUrlVisits(VisitsStats);
|
|
||||||
|
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
<ShortUrlVisits
|
<ShortUrlVisits
|
||||||
{...Mock.all<ShortUrlVisitsProps>()}
|
{...Mock.all<ShortUrlVisitsProps>()}
|
||||||
|
@ -34,7 +32,7 @@ describe('<ShortUrlVisits />', () => {
|
||||||
history={history}
|
history={history}
|
||||||
shortUrlVisits={Mock.of<ShortUrlVisitsState>({ loading: true, visits: [] })}
|
shortUrlVisits={Mock.of<ShortUrlVisitsState>({ loading: true, visits: [] })}
|
||||||
shortUrlDetail={Mock.all<ShortUrlDetail>()}
|
shortUrlDetail={Mock.all<ShortUrlDetail>()}
|
||||||
cancelGetShortUrlVisits={identity}
|
cancelGetShortUrlVisits={() => {}}
|
||||||
/>,
|
/>,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,22 +1,25 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
import Moment from 'react-moment';
|
import Moment from 'react-moment';
|
||||||
import { ExternalLink } from 'react-external-link';
|
import { ExternalLink } from 'react-external-link';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
import ShortUrlVisitsHeader from '../../src/visits/ShortUrlVisitsHeader';
|
import ShortUrlVisitsHeader from '../../src/visits/ShortUrlVisitsHeader';
|
||||||
|
import { ShortUrlDetail } from '../../src/visits/reducers/shortUrlDetail';
|
||||||
|
import { ShortUrlVisits } from '../../src/visits/reducers/shortUrlVisits';
|
||||||
|
|
||||||
describe('<ShortUrlVisitsHeader />', () => {
|
describe('<ShortUrlVisitsHeader />', () => {
|
||||||
let wrapper;
|
let wrapper: ShallowWrapper;
|
||||||
const shortUrlDetail = {
|
const shortUrlDetail = Mock.of<ShortUrlDetail>({
|
||||||
shortUrl: {
|
shortUrl: {
|
||||||
shortUrl: 'https://doma.in/abc123',
|
shortUrl: 'https://doma.in/abc123',
|
||||||
longUrl: 'https://foo.bar/bar/foo',
|
longUrl: 'https://foo.bar/bar/foo',
|
||||||
dateCreated: '2018-01-01T10:00:00+01:00',
|
dateCreated: '2018-01-01T10:00:00+01:00',
|
||||||
},
|
},
|
||||||
loading: false,
|
loading: false,
|
||||||
};
|
});
|
||||||
const shortUrlVisits = {
|
const shortUrlVisits = Mock.of<ShortUrlVisits>({
|
||||||
visits: [{}, {}, {}],
|
visits: [{}, {}, {}],
|
||||||
};
|
});
|
||||||
const goBack = jest.fn();
|
const goBack = jest.fn();
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
@ -29,12 +32,12 @@ describe('<ShortUrlVisitsHeader />', () => {
|
||||||
it('shows when the URL was created', () => {
|
it('shows when the URL was created', () => {
|
||||||
const moment = wrapper.find(Moment).first();
|
const moment = wrapper.find(Moment).first();
|
||||||
|
|
||||||
expect(moment.prop('children')).toEqual(shortUrlDetail.shortUrl.dateCreated);
|
expect(moment.prop('children')).toEqual(shortUrlDetail.shortUrl?.dateCreated);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('shows the long URL', () => {
|
it('shows the long URL', () => {
|
||||||
const longUrlLink = wrapper.find(ExternalLink).last();
|
const longUrlLink = wrapper.find(ExternalLink).last();
|
||||||
|
|
||||||
expect(longUrlLink.prop('href')).toEqual(shortUrlDetail.shortUrl.longUrl);
|
expect(longUrlLink.prop('href')).toEqual(shortUrlDetail.shortUrl?.longUrl);
|
||||||
});
|
});
|
||||||
});
|
});
|
|
@ -1,42 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import { shallow } from 'enzyme';
|
|
||||||
import { identity } from 'ramda';
|
|
||||||
import createTagVisits from '../../src/visits/TagVisits';
|
|
||||||
import TagVisitsHeader from '../../src/visits/TagVisitsHeader';
|
|
||||||
|
|
||||||
describe('<TagVisits />', () => {
|
|
||||||
let wrapper;
|
|
||||||
const getTagVisitsMock = jest.fn();
|
|
||||||
const match = {
|
|
||||||
params: { tag: 'foo' },
|
|
||||||
};
|
|
||||||
const history = {
|
|
||||||
goBack: jest.fn(),
|
|
||||||
};
|
|
||||||
const VisitsStats = jest.fn();
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
const TagVisits = createTagVisits(VisitsStats, {});
|
|
||||||
|
|
||||||
wrapper = shallow(
|
|
||||||
<TagVisits
|
|
||||||
getTagVisits={getTagVisitsMock}
|
|
||||||
match={match}
|
|
||||||
history={history}
|
|
||||||
tagVisits={{ loading: true, visits: [] }}
|
|
||||||
cancelGetTagVisits={identity}
|
|
||||||
/>,
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => wrapper.unmount());
|
|
||||||
afterEach(jest.resetAllMocks);
|
|
||||||
|
|
||||||
it('renders visit stats and visits header', () => {
|
|
||||||
const visitStats = wrapper.find(VisitsStats);
|
|
||||||
const visitHeader = wrapper.find(TagVisitsHeader);
|
|
||||||
|
|
||||||
expect(visitStats).toHaveLength(1);
|
|
||||||
expect(visitHeader).toHaveLength(1);
|
|
||||||
});
|
|
||||||
});
|
|
47
test/visits/TagVisits.test.tsx
Normal file
47
test/visits/TagVisits.test.tsx
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import React from 'react';
|
||||||
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
|
import { History } from 'history';
|
||||||
|
import { match } from 'react-router'; // eslint-disable-line @typescript-eslint/no-unused-vars
|
||||||
|
import createTagVisits, { TagVisitsProps } from '../../src/visits/TagVisits';
|
||||||
|
import TagVisitsHeader from '../../src/visits/TagVisitsHeader';
|
||||||
|
import ColorGenerator from '../../src/utils/services/ColorGenerator';
|
||||||
|
import { TagVisits as TagVisitsStats } from '../../src/visits/reducers/tagVisits';
|
||||||
|
import VisitsStats from '../../src/visits/VisitsStats';
|
||||||
|
|
||||||
|
describe('<TagVisits />', () => {
|
||||||
|
let wrapper: ShallowWrapper;
|
||||||
|
const getTagVisitsMock = jest.fn();
|
||||||
|
const match = Mock.of<match<{ tag: string }>>({
|
||||||
|
params: { tag: 'foo' },
|
||||||
|
});
|
||||||
|
const history = Mock.of<History>({
|
||||||
|
goBack: jest.fn(),
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
const TagVisits = createTagVisits(Mock.of<ColorGenerator>());
|
||||||
|
|
||||||
|
wrapper = shallow(
|
||||||
|
<TagVisits
|
||||||
|
{...Mock.all<TagVisitsProps>()}
|
||||||
|
getTagVisits={getTagVisitsMock}
|
||||||
|
match={match}
|
||||||
|
history={history}
|
||||||
|
tagVisits={Mock.of<TagVisitsStats>({ loading: true, visits: [] })}
|
||||||
|
cancelGetTagVisits={() => {}}
|
||||||
|
/>,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => wrapper.unmount());
|
||||||
|
afterEach(jest.resetAllMocks);
|
||||||
|
|
||||||
|
it('renders visit stats and visits header', () => {
|
||||||
|
const visitStats = wrapper.find(VisitsStats);
|
||||||
|
const visitHeader = wrapper.find(TagVisitsHeader);
|
||||||
|
|
||||||
|
expect(visitStats).toHaveLength(1);
|
||||||
|
expect(visitHeader).toHaveLength(1);
|
||||||
|
});
|
||||||
|
});
|
|
@ -1,19 +1,22 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
import Tag from '../../src/tags/helpers/Tag';
|
import Tag from '../../src/tags/helpers/Tag';
|
||||||
import TagVisitsHeader from '../../src/visits/TagVisitsHeader';
|
import TagVisitsHeader from '../../src/visits/TagVisitsHeader';
|
||||||
|
import { TagVisits } from '../../src/visits/reducers/tagVisits';
|
||||||
|
import ColorGenerator from '../../src/utils/services/ColorGenerator';
|
||||||
|
|
||||||
describe('<TagVisitsHeader />', () => {
|
describe('<TagVisitsHeader />', () => {
|
||||||
let wrapper;
|
let wrapper: ShallowWrapper;
|
||||||
const tagVisits = {
|
const tagVisits = Mock.of<TagVisits>({
|
||||||
tag: 'foo',
|
tag: 'foo',
|
||||||
visits: [{}, {}, {}],
|
visits: [{}, {}, {}],
|
||||||
};
|
});
|
||||||
const goBack = jest.fn();
|
const goBack = jest.fn();
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
<TagVisitsHeader tagVisits={tagVisits} goBack={goBack} colorGenerator={{}} />,
|
<TagVisitsHeader tagVisits={tagVisits} goBack={goBack} colorGenerator={Mock.all<ColorGenerator>()} />,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
afterEach(() => wrapper.unmount());
|
afterEach(() => wrapper.unmount());
|
|
@ -1,10 +1,12 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
import VisitsHeader from '../../src/visits/VisitsHeader';
|
import VisitsHeader from '../../src/visits/VisitsHeader';
|
||||||
|
import { Visit } from '../../src/visits/types';
|
||||||
|
|
||||||
describe('<VisitsHeader />', () => {
|
describe('<VisitsHeader />', () => {
|
||||||
let wrapper;
|
let wrapper: ShallowWrapper;
|
||||||
const visits = [{}, {}, {}];
|
const visits = [ Mock.all<Visit>(), Mock.all<Visit>(), Mock.all<Visit>() ];
|
||||||
const title = 'My header title';
|
const title = 'My header title';
|
||||||
const goBack = jest.fn();
|
const goBack = jest.fn();
|
||||||
|
|
|
@ -1,36 +1,34 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
import { identity } from 'ramda';
|
|
||||||
import { Card, Progress } from 'reactstrap';
|
import { Card, Progress } from 'reactstrap';
|
||||||
import createVisitStats from '../../src/visits/VisitsStats';
|
import { Mock } from 'ts-mockery';
|
||||||
|
import VisitStats from '../../src/visits/VisitsStats';
|
||||||
import Message from '../../src/utils/Message';
|
import Message from '../../src/utils/Message';
|
||||||
import GraphCard from '../../src/visits/helpers/GraphCard';
|
import GraphCard from '../../src/visits/helpers/GraphCard';
|
||||||
import SortableBarGraph from '../../src/visits/helpers/SortableBarGraph';
|
import SortableBarGraph from '../../src/visits/helpers/SortableBarGraph';
|
||||||
import DateRangeRow from '../../src/utils/DateRangeRow';
|
import DateRangeRow from '../../src/utils/DateRangeRow';
|
||||||
|
import { Visit, VisitsInfo } from '../../src/visits/types';
|
||||||
|
|
||||||
describe('<VisitStats />', () => {
|
describe('<VisitStats />', () => {
|
||||||
let wrapper;
|
const visits = [ Mock.all<Visit>(), Mock.all<Visit>(), Mock.all<Visit>() ];
|
||||||
const processStatsFromVisits = () => (
|
|
||||||
{ os: {}, browsers: {}, referrers: {}, countries: {}, cities: {}, citiesForMap: {} }
|
let wrapper: ShallowWrapper;
|
||||||
);
|
|
||||||
const getVisitsMock = jest.fn();
|
const getVisitsMock = jest.fn();
|
||||||
|
|
||||||
const createComponent = (visitsInfo) => {
|
const createComponent = (visitsInfo: Partial<VisitsInfo>) => {
|
||||||
const VisitStats = createVisitStats({ processStatsFromVisits, normalizeVisits: identity }, () => '');
|
|
||||||
|
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
<VisitStats
|
<VisitStats
|
||||||
getVisits={getVisitsMock}
|
getVisits={getVisitsMock}
|
||||||
visitsInfo={visitsInfo}
|
visitsInfo={Mock.of<VisitsInfo>(visitsInfo)}
|
||||||
cancelGetVisits={identity}
|
cancelGetVisits={() => {}}
|
||||||
matchMedia={() => ({ matches: false })}
|
matchMedia={() => Mock.of<MediaQueryList>({ matches: false })}
|
||||||
/>,
|
/>,
|
||||||
);
|
);
|
||||||
|
|
||||||
return wrapper;
|
return wrapper;
|
||||||
};
|
};
|
||||||
|
|
||||||
afterEach(() => wrapper && wrapper.unmount());
|
afterEach(() => wrapper?.unmount());
|
||||||
|
|
||||||
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: [] });
|
||||||
|
@ -70,7 +68,7 @@ describe('<VisitStats />', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('renders all graphics when visits are properly loaded', () => {
|
it('renders all graphics when visits are properly loaded', () => {
|
||||||
const wrapper = createComponent({ loading: false, error: false, visits: [{}, {}, {}] });
|
const wrapper = createComponent({ loading: false, error: false, visits });
|
||||||
const graphs = wrapper.find(GraphCard);
|
const graphs = wrapper.find(GraphCard);
|
||||||
const sortableBarGraphs = wrapper.find(SortableBarGraph);
|
const sortableBarGraphs = wrapper.find(SortableBarGraph);
|
||||||
|
|
||||||
|
@ -78,7 +76,7 @@ describe('<VisitStats />', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('reloads visits when selected dates change', () => {
|
it('reloads visits when selected dates change', () => {
|
||||||
const wrapper = createComponent({ loading: false, error: false, visits: [{}, {}, {}] });
|
const wrapper = createComponent({ loading: false, error: false, visits });
|
||||||
const dateRange = wrapper.find(DateRangeRow);
|
const dateRange = wrapper.find(DateRangeRow);
|
||||||
|
|
||||||
dateRange.simulate('startDateChange', '2016-01-01T00:00:00+01:00');
|
dateRange.simulate('startDateChange', '2016-01-01T00:00:00+01:00');
|
||||||
|
@ -90,7 +88,7 @@ describe('<VisitStats />', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('holds the map button content generator on cities graph extraHeaderContent', () => {
|
it('holds the map button content generator on cities graph extraHeaderContent', () => {
|
||||||
const wrapper = createComponent({ loading: false, error: false, visits: [{}, {}, {}] });
|
const wrapper = createComponent({ loading: false, error: false, visits });
|
||||||
const citiesGraph = wrapper.find(SortableBarGraph).find('[title="Cities"]');
|
const citiesGraph = wrapper.find(SortableBarGraph).find('[title="Cities"]');
|
||||||
const extraHeaderContent = citiesGraph.prop('extraHeaderContent');
|
const extraHeaderContent = citiesGraph.prop('extraHeaderContent');
|
||||||
|
|
|
@ -1,15 +1,17 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { shallow } from 'enzyme';
|
import { shallow, ShallowWrapper } from 'enzyme';
|
||||||
|
import { Mock } from 'ts-mockery';
|
||||||
import VisitsTable from '../../src/visits/VisitsTable';
|
import VisitsTable from '../../src/visits/VisitsTable';
|
||||||
import { rangeOf } from '../../src/utils/utils';
|
import { rangeOf } from '../../src/utils/utils';
|
||||||
import SimplePaginator from '../../src/common/SimplePaginator';
|
import SimplePaginator from '../../src/common/SimplePaginator';
|
||||||
import SearchField from '../../src/utils/SearchField';
|
import SearchField from '../../src/utils/SearchField';
|
||||||
|
import { NormalizedVisit } from '../../src/visits/types';
|
||||||
|
|
||||||
describe('<VisitsTable />', () => {
|
describe('<VisitsTable />', () => {
|
||||||
const matchMedia = () => ({ matches: false });
|
const matchMedia = () => Mock.of<MediaQueryList>({ matches: false });
|
||||||
const setSelectedVisits = jest.fn();
|
const setSelectedVisits = jest.fn();
|
||||||
let wrapper;
|
let wrapper: ShallowWrapper;
|
||||||
const createWrapper = (visits, selectedVisits = []) => {
|
const createWrapper = (visits: NormalizedVisit[], selectedVisits: NormalizedVisit[] = []) => {
|
||||||
wrapper = shallow(
|
wrapper = shallow(
|
||||||
<VisitsTable
|
<VisitsTable
|
||||||
visits={visits}
|
visits={visits}
|
||||||
|
@ -22,10 +24,8 @@ describe('<VisitsTable />', () => {
|
||||||
return wrapper;
|
return wrapper;
|
||||||
};
|
};
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(jest.resetAllMocks);
|
||||||
jest.resetAllMocks();
|
afterEach(() => wrapper?.unmount());
|
||||||
wrapper && wrapper.unmount();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('renders columns as expected', () => {
|
it('renders columns as expected', () => {
|
||||||
const wrapper = createWrapper([]);
|
const wrapper = createWrapper([]);
|
||||||
|
@ -55,7 +55,9 @@ describe('<VisitsTable />', () => {
|
||||||
[ 60, 3 ],
|
[ 60, 3 ],
|
||||||
[ 115, 6 ],
|
[ 115, 6 ],
|
||||||
])('renders the expected amount of pages', (visitsCount, expectedAmountOfPages) => {
|
])('renders the expected amount of pages', (visitsCount, expectedAmountOfPages) => {
|
||||||
const wrapper = createWrapper(rangeOf(visitsCount, () => ({ browser: '', date: '', referer: '' })));
|
const wrapper = createWrapper(
|
||||||
|
rangeOf(visitsCount, () => Mock.of<NormalizedVisit>({ browser: '', date: '', referer: '' })),
|
||||||
|
);
|
||||||
const tr = wrapper.find('tbody').find('tr');
|
const tr = wrapper.find('tbody').find('tr');
|
||||||
const paginator = wrapper.find(SimplePaginator);
|
const paginator = wrapper.find(SimplePaginator);
|
||||||
|
|
||||||
|
@ -66,7 +68,9 @@ describe('<VisitsTable />', () => {
|
||||||
it.each(
|
it.each(
|
||||||
rangeOf(20, (value) => [ value ]),
|
rangeOf(20, (value) => [ value ]),
|
||||||
)('does not render footer when there is only one page to render', (visitsCount) => {
|
)('does not render footer when there is only one page to render', (visitsCount) => {
|
||||||
const wrapper = createWrapper(rangeOf(visitsCount, () => ({ browser: '', date: '', referer: '' })));
|
const wrapper = createWrapper(
|
||||||
|
rangeOf(visitsCount, () => Mock.of<NormalizedVisit>({ browser: '', date: '', referer: '' })),
|
||||||
|
);
|
||||||
const tr = wrapper.find('tbody').find('tr');
|
const tr = wrapper.find('tbody').find('tr');
|
||||||
const paginator = wrapper.find(SimplePaginator);
|
const paginator = wrapper.find(SimplePaginator);
|
||||||
|
|
||||||
|
@ -75,7 +79,7 @@ describe('<VisitsTable />', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('selected rows are highlighted', () => {
|
it('selected rows are highlighted', () => {
|
||||||
const visits = rangeOf(10, () => ({ browser: '', date: '', referer: '' }));
|
const visits = rangeOf(10, () => Mock.of<NormalizedVisit>({ browser: '', date: '', referer: '' }));
|
||||||
const wrapper = createWrapper(
|
const wrapper = createWrapper(
|
||||||
visits,
|
visits,
|
||||||
[ visits[1], visits[2] ],
|
[ visits[1], visits[2] ],
|
||||||
|
@ -98,7 +102,7 @@ describe('<VisitsTable />', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('orders visits when column is clicked', () => {
|
it('orders visits when column is clicked', () => {
|
||||||
const wrapper = createWrapper(rangeOf(9, (index) => ({
|
const wrapper = createWrapper(rangeOf(9, (index) => Mock.of<NormalizedVisit>({
|
||||||
browser: '',
|
browser: '',
|
||||||
date: `${9 - index}`,
|
date: `${9 - index}`,
|
||||||
referer: `${index}`,
|
referer: `${index}`,
|
||||||
|
@ -118,8 +122,8 @@ describe('<VisitsTable />', () => {
|
||||||
|
|
||||||
it('filters list when writing in search box', () => {
|
it('filters list when writing in search box', () => {
|
||||||
const wrapper = createWrapper([
|
const wrapper = createWrapper([
|
||||||
...rangeOf(7, () => ({ browser: 'aaa', date: 'aaa', referer: 'aaa' })),
|
...rangeOf(7, () => Mock.of<NormalizedVisit>({ browser: 'aaa', date: 'aaa', referer: 'aaa' })),
|
||||||
...rangeOf(2, () => ({ browser: 'bbb', date: 'bbb', referer: 'bbb' })),
|
...rangeOf(2, () => Mock.of<NormalizedVisit>({ browser: 'bbb', date: 'bbb', referer: 'bbb' })),
|
||||||
]);
|
]);
|
||||||
const searchField = wrapper.find(SearchField);
|
const searchField = wrapper.find(SearchField);
|
||||||
|
|
Loading…
Add table
Reference in a new issue