mirror of
https://github.com/kremalicious/umami.git
synced 2024-11-22 01:46:58 +01:00
Updated timezone hook, fixed chart rendering, added icons.
This commit is contained in:
parent
5a2330ba2a
commit
291562f6c2
3
src/app/(main)/profile/DateRangeSetting.module.css
Normal file
3
src/app/(main)/profile/DateRangeSetting.module.css
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
.field {
|
||||||
|
width: 200px;
|
||||||
|
}
|
@ -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}
|
||||||
|
@ -1,3 +1,7 @@
|
|||||||
|
.dropdown {
|
||||||
|
width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
div.menu {
|
div.menu {
|
||||||
max-height: 300px;
|
max-height: 300px;
|
||||||
width: 300px;
|
width: 300px;
|
||||||
|
@ -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}
|
||||||
|
@ -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}
|
||||||
/>
|
/>
|
||||||
|
@ -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 = {
|
||||||
|
@ -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();
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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 });
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user