Updated timezone hook, fixed chart rendering, added icons.

This commit is contained in:
Mike Cao 2024-03-29 16:04:39 -07:00
parent 5a2330ba2a
commit 291562f6c2
14 changed files with 94 additions and 65 deletions

View File

@ -0,0 +1,3 @@
.field {
width: 200px;
}

View File

@ -3,6 +3,7 @@ import { Button, Flexbox } from 'react-basics';
import { useDateRange, useMessages } from 'components/hooks'; import { useDateRange, useMessages } from 'components/hooks';
import { DEFAULT_DATE_RANGE } from 'lib/constants'; import { DEFAULT_DATE_RANGE } from 'lib/constants';
import { DateRange } from 'lib/types'; import { DateRange } from 'lib/types';
import styles from './DateRangeSetting.module.css';
export function DateRangeSetting() { export function DateRangeSetting() {
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
@ -13,8 +14,9 @@ export function DateRangeSetting() {
const handleReset = () => setDateRange(DEFAULT_DATE_RANGE); const handleReset = () => setDateRange(DEFAULT_DATE_RANGE);
return ( return (
<Flexbox gap={10}> <Flexbox gap={10} width={300}>
<DateFilter <DateFilter
className={styles.field}
value={value} value={value}
startDate={dateRange.startDate} startDate={dateRange.startDate}
endDate={dateRange.endDate} endDate={dateRange.endDate}

View File

@ -1,3 +1,7 @@
.dropdown {
width: 200px;
}
div.menu { div.menu {
max-height: 300px; max-height: 300px;
width: 300px; width: 300px;

View File

@ -1,26 +1,29 @@
import { useState } from 'react'; import { useState } from 'react';
import { Dropdown, Item, Button, Flexbox } from 'react-basics'; import { Dropdown, Item, Button, Flexbox } from 'react-basics';
import { listTimeZones } from 'timezone-support'; import moment from 'moment-timezone';
import { useTimezone, useMessages } from 'components/hooks'; import { useTimezone, useMessages } from 'components/hooks';
import { getTimezone } from 'lib/date'; import { getTimezone } from 'lib/date';
import styles from './TimezoneSetting.module.css'; import styles from './TimezoneSetting.module.css';
const timezones = moment.tz.names();
export function TimezoneSetting() { export function TimezoneSetting() {
const [search, setSearch] = useState(''); const [search, setSearch] = useState('');
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
const [timezone, saveTimezone] = useTimezone(); const { timezone, saveTimezone } = useTimezone();
const options = search const options = search
? listTimeZones().filter(n => n.toLowerCase().includes(search.toLowerCase())) ? timezones.filter(n => n.toLowerCase().includes(search.toLowerCase()))
: listTimeZones(); : timezones;
const handleReset = () => saveTimezone(getTimezone()); const handleReset = () => saveTimezone(getTimezone());
return ( return (
<Flexbox gap={10}> <Flexbox gap={10}>
<Dropdown <Dropdown
className={styles.dropdown}
items={options} items={options}
value={timezone} value={timezone}
onChange={saveTimezone} onChange={(value: any) => saveTimezone(value)}
menuProps={{ className: styles.menu }} menuProps={{ className: styles.menu }}
allowSearch={true} allowSearch={true}
onSearch={setSearch} onSearch={setSearch}

View File

@ -16,7 +16,8 @@ export function Favicon({ domain, ...props }) {
<img <img
className={styles.favicon} className={styles.favicon}
src={`https://icons.duckduckgo.com/ip3/${hostName}.ico`} src={`https://icons.duckduckgo.com/ip3/${hostName}.ico`}
height="16" width={16}
height={16}
alt="" alt=""
{...props} {...props}
/> />

View File

@ -8,7 +8,7 @@ export function useReport(reportId: string, defaultParameters: { [key: string]:
const [report, setReport] = useState(null); const [report, setReport] = useState(null);
const [isRunning, setIsRunning] = useState(false); const [isRunning, setIsRunning] = useState(false);
const { get, post } = useApi(); const { get, post } = useApi();
const [timezone] = useTimezone(); const { timezone } = useTimezone();
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
const baseParameters = { const baseParameters = {

View File

@ -4,7 +4,7 @@ export function useWebsitePageviews(websiteId: string, options?: { [key: string]
const { get, useQuery } = useApi(); const { get, useQuery } = useApi();
const [dateRange] = useDateRange(websiteId); const [dateRange] = useDateRange(websiteId);
const { startDate, endDate, unit } = dateRange; const { startDate, endDate, unit } = dateRange;
const [timezone] = useTimezone(); const { timezone } = useTimezone();
const { const {
query: { url, referrer, os, browser, device, country, region, city, title }, query: { url, referrer, os, browser, device, country, region, city, title },
} = useNavigation(); } = useNavigation();

View File

@ -1,4 +1,4 @@
import { useEffect } from 'react'; import { useEffect, useMemo } from 'react';
import useStore, { setTheme } from 'store/app'; import useStore, { setTheme } from 'store/app';
import { getItem, setItem } from 'next-basics'; import { getItem, setItem } from 'next-basics';
import { DEFAULT_THEME, THEME_COLORS, THEME_CONFIG } from 'lib/constants'; import { DEFAULT_THEME, THEME_COLORS, THEME_CONFIG } from 'lib/constants';
@ -10,38 +10,40 @@ export function useTheme() {
const theme = useStore(selector) || getItem(THEME_CONFIG) || DEFAULT_THEME; const theme = useStore(selector) || getItem(THEME_CONFIG) || DEFAULT_THEME;
const primaryColor = colord(THEME_COLORS[theme].primary); const primaryColor = colord(THEME_COLORS[theme].primary);
const colors = { const colors = useMemo(() => {
theme: { return {
...THEME_COLORS[theme], theme: {
}, ...THEME_COLORS[theme],
chart: {
text: THEME_COLORS[theme].gray700,
line: THEME_COLORS[theme].gray200,
views: {
hoverBackgroundColor: primaryColor.alpha(0.7).toRgbString(),
backgroundColor: primaryColor.alpha(0.4).toRgbString(),
borderColor: primaryColor.alpha(0.7).toRgbString(),
hoverBorderColor: primaryColor.toRgbString(),
}, },
visitors: { chart: {
hoverBackgroundColor: primaryColor.alpha(0.9).toRgbString(), text: THEME_COLORS[theme].gray700,
backgroundColor: primaryColor.alpha(0.6).toRgbString(), line: THEME_COLORS[theme].gray200,
borderColor: primaryColor.alpha(0.9).toRgbString(), views: {
hoverBorderColor: primaryColor.toRgbString(), hoverBackgroundColor: primaryColor.alpha(0.7).toRgbString(),
backgroundColor: primaryColor.alpha(0.4).toRgbString(),
borderColor: primaryColor.alpha(0.7).toRgbString(),
hoverBorderColor: primaryColor.toRgbString(),
},
visitors: {
hoverBackgroundColor: primaryColor.alpha(0.9).toRgbString(),
backgroundColor: primaryColor.alpha(0.6).toRgbString(),
borderColor: primaryColor.alpha(0.9).toRgbString(),
hoverBorderColor: primaryColor.toRgbString(),
},
}, },
}, map: {
map: { baseColor: THEME_COLORS[theme].primary,
baseColor: THEME_COLORS[theme].primary, fillColor: THEME_COLORS[theme].gray100,
fillColor: THEME_COLORS[theme].gray100, strokeColor: THEME_COLORS[theme].primary,
strokeColor: THEME_COLORS[theme].primary, hoverColor: THEME_COLORS[theme].primary,
hoverColor: THEME_COLORS[theme].primary, },
}, };
}; }, [theme]);
function saveTheme(value) { const saveTheme = (value: string) => {
setItem(THEME_CONFIG, value); setItem(THEME_CONFIG, value);
setTheme(value); setTheme(value);
} };
useEffect(() => { useEffect(() => {
document.body.setAttribute('data-theme', theme); document.body.setAttribute('data-theme', theme);

View File

@ -1,20 +1,18 @@
import { useState, useCallback } from 'react'; import { setItem } from 'next-basics';
import { getTimezone } from 'lib/date';
import { getItem, setItem } from 'next-basics';
import { TIMEZONE_CONFIG } from 'lib/constants'; import { TIMEZONE_CONFIG } from 'lib/constants';
import useStore, { setTimezone } from 'store/app';
const selector = (state: { timezone: string }) => state.timezone;
export function useTimezone() { export function useTimezone() {
const [timezone, setTimezone] = useState(getItem(TIMEZONE_CONFIG) || getTimezone()); const timezone = useStore(selector);
const saveTimezone = useCallback( const saveTimezone = (value: string) => {
(value: string) => { setItem(TIMEZONE_CONFIG, value);
setItem(TIMEZONE_CONFIG, value); setTimezone(value);
setTimezone(value); };
},
[setTimezone],
);
return [timezone, saveTimezone]; return { timezone, saveTimezone };
} }
export default useTimezone; export default useTimezone;

View File

@ -22,7 +22,7 @@ export interface EventsChartProps {
export function EventsChart({ websiteId, className, token }: EventsChartProps) { export function EventsChart({ websiteId, className, token }: EventsChartProps) {
const [{ startDate, endDate, unit, offset }] = useDateRange(websiteId); const [{ startDate, endDate, unit, offset }] = useDateRange(websiteId);
const { locale } = useLocale(); const { locale } = useLocale();
const [timezone] = useTimezone(); const { timezone } = useTimezone();
const { const {
query: { url, event }, query: { url, event },
} = useNavigation(); } = useNavigation();

View File

@ -1,18 +1,23 @@
import MetricsTable, { MetricsTableProps } from './MetricsTable'; import MetricsTable, { MetricsTableProps } from './MetricsTable';
import FilterLink from 'components/common/FilterLink'; import FilterLink from 'components/common/FilterLink';
import Favicon from 'components/common/Favicon';
import { useMessages } from 'components/hooks'; import { useMessages } from 'components/hooks';
import { Flexbox } from 'react-basics';
export function ReferrersTable(props: MetricsTableProps) { export function ReferrersTable(props: MetricsTableProps) {
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
const renderLink = ({ x: referrer }) => { const renderLink = ({ x: referrer }) => {
return ( return (
<FilterLink <Flexbox alignItems="center">
id="referrer" <Favicon domain={referrer} />
value={referrer} <FilterLink
externalUrl={`https://${referrer}`} id="referrer"
label={!referrer && formatMessage(labels.none)} value={referrer}
/> externalUrl={`https://${referrer}`}
label={!referrer && formatMessage(labels.none)}
/>
</Flexbox>
); );
}; };

View File

@ -20,16 +20,17 @@ import * as yup from 'yup';
export interface CollectRequestBody { export interface CollectRequestBody {
payload: { payload: {
data: { [key: string]: any };
hostname: string;
ip: string;
language: string;
referrer: string;
screen: string;
title: string;
url: string;
website: string; website: string;
name: string; data?: { [key: string]: any };
hostname?: string;
ip?: string;
language?: string;
name?: string;
referrer?: string;
screen?: string;
tag?: string;
title?: string;
url: string;
}; };
type: CollectionType; type: CollectionType;
} }
@ -72,6 +73,7 @@ const schema = {
url: yup.string(), url: yup.string(),
website: yup.string().uuid().required(), website: yup.string().uuid().required(),
name: yup.string().max(50), name: yup.string().max(50),
tag: yup.string().max(50),
}) })
.required(), .required(),
type: yup type: yup

View File

@ -6,8 +6,10 @@ import {
DEFAULT_THEME, DEFAULT_THEME,
LOCALE_CONFIG, LOCALE_CONFIG,
THEME_CONFIG, THEME_CONFIG,
TIMEZONE_CONFIG,
} from 'lib/constants'; } from 'lib/constants';
import { getItem } from 'next-basics'; import { getItem } from 'next-basics';
import { getTimezone } from 'lib/date';
function getDefaultTheme() { function getDefaultTheme() {
return typeof window !== 'undefined' return typeof window !== 'undefined'
@ -20,6 +22,7 @@ function getDefaultTheme() {
const initialState = { const initialState = {
locale: getItem(LOCALE_CONFIG) || DEFAULT_LOCALE, locale: getItem(LOCALE_CONFIG) || DEFAULT_LOCALE,
theme: getItem(THEME_CONFIG) || getDefaultTheme() || DEFAULT_THEME, theme: getItem(THEME_CONFIG) || getDefaultTheme() || DEFAULT_THEME,
timezone: getItem(TIMEZONE_CONFIG) || getTimezone(),
dateRange: getItem(DATE_RANGE_CONFIG) || DEFAULT_DATE_RANGE, dateRange: getItem(DATE_RANGE_CONFIG) || DEFAULT_DATE_RANGE,
shareToken: null, shareToken: null,
user: null, user: null,
@ -32,6 +35,10 @@ export function setTheme(theme: string) {
store.setState({ theme }); store.setState({ theme });
} }
export function setTimezone(timezone: string) {
store.setState({ timezone });
}
export function setLocale(locale: string) { export function setLocale(locale: string) {
store.setState({ locale }); store.setState({ locale });
} }

View File

@ -18,6 +18,7 @@
const attr = currentScript.getAttribute.bind(currentScript); const attr = currentScript.getAttribute.bind(currentScript);
const website = attr(_data + 'website-id'); const website = attr(_data + 'website-id');
const hostUrl = attr(_data + 'host-url'); const hostUrl = attr(_data + 'host-url');
const tag = attr(_data + 'tag');
const autoTrack = attr(_data + 'auto-track') !== _false; const autoTrack = attr(_data + 'auto-track') !== _false;
const excludeSearch = attr(_data + 'exclude-search') === _true; const excludeSearch = attr(_data + 'exclude-search') === _true;
const domain = attr(_data + 'domains') || ''; const domain = attr(_data + 'domains') || '';
@ -216,6 +217,7 @@
...getPayload(), ...getPayload(),
name: obj, name: obj,
data: typeof data === 'object' ? data : undefined, data: typeof data === 'object' ? data : undefined,
tag,
}); });
} else if (typeof obj === 'object') { } else if (typeof obj === 'object') {
return send(obj); return send(obj);