diff --git a/.gitignore b/.gitignore
index 715ff703..32d3cbce 100644
--- a/.gitignore
+++ b/.gitignore
@@ -23,6 +23,7 @@
.DS_Store
.idea
*.iml
+*.log
.vscode/*
# debug
diff --git a/components/common/Dot.js b/components/common/Dot.js
index 3f424820..d5dcf914 100644
--- a/components/common/Dot.js
+++ b/components/common/Dot.js
@@ -4,12 +4,14 @@ import styles from './Dot.module.css';
export default function Dot({ color, size, className }) {
return (
-
+
);
}
diff --git a/components/common/Dot.module.css b/components/common/Dot.module.css
index 9081dc5c..258d6e87 100644
--- a/components/common/Dot.module.css
+++ b/components/common/Dot.module.css
@@ -1,9 +1,14 @@
+.wrapper {
+ background: var(--gray50);
+ margin-right: 10px;
+ border-radius: 100%;
+}
+
.dot {
background: var(--green400);
width: 10px;
height: 10px;
border-radius: 100%;
- margin-right: 10px;
}
.dot.small {
diff --git a/components/common/Favicon.js b/components/common/Favicon.js
new file mode 100644
index 00000000..07ec696c
--- /dev/null
+++ b/components/common/Favicon.js
@@ -0,0 +1,21 @@
+import React from 'react';
+import styles from './Favicon.module.css';
+
+function getHostName(url) {
+ const match = url.match(/^(?:https?:\/\/)?(?:[^@\n]+@)?(?:www\.)?([^:/\n?=]+)/im);
+ return match && match.length > 1 ? match[1] : null;
+}
+
+export default function Favicon({ domain, ...props }) {
+ const hostName = domain ? getHostName(domain) : null;
+
+ return hostName ? (
+
+ ) : null;
+}
diff --git a/components/common/Favicon.module.css b/components/common/Favicon.module.css
new file mode 100644
index 00000000..82c85c42
--- /dev/null
+++ b/components/common/Favicon.module.css
@@ -0,0 +1,3 @@
+.favicon {
+ margin-right: 8px;
+}
diff --git a/components/common/NoData.module.css b/components/common/NoData.module.css
index d1c712eb..82f9c3ee 100644
--- a/components/common/NoData.module.css
+++ b/components/common/NoData.module.css
@@ -1,5 +1,6 @@
.container {
color: var(--gray500);
+ font-size: var(--font-size-normal);
position: absolute;
top: 50%;
left: 50%;
diff --git a/components/metrics/BarChart.js b/components/metrics/BarChart.js
index a31a6f40..f42cf73f 100644
--- a/components/metrics/BarChart.js
+++ b/components/metrics/BarChart.js
@@ -1,13 +1,15 @@
import React, { useState, useRef, useEffect } from 'react';
-import ReactTooltip from 'react-tooltip';
import classNames from 'classnames';
import ChartJS from 'chart.js';
+import Legend from 'components/metrics/Legend';
import { formatLongNumber } from 'lib/format';
import { dateFormat } from 'lib/lang';
import useLocale from 'hooks/useLocale';
import useTheme from 'hooks/useTheme';
import { DEFAUL_CHART_HEIGHT, DEFAULT_ANIMATION_DURATION, THEME_COLORS } from 'lib/constants';
import styles from './BarChart.module.css';
+import ChartTooltip from './ChartTooltip';
+import useForceUpdate from '../../hooks/useForceUpdate';
export default function BarChart({
chartId,
@@ -27,6 +29,8 @@ export default function BarChart({
const [tooltip, setTooltip] = useState(null);
const [locale] = useLocale();
const [theme] = useTheme();
+ const forceUpdate = useForceUpdate();
+
const colors = {
text: THEME_COLORS[theme].gray700,
line: THEME_COLORS[theme].gray200,
@@ -111,9 +115,7 @@ export default function BarChart({
responsiveAnimationDuration: 0,
maintainAspectRatio: false,
legend: {
- labels: {
- fontColor: colors.text,
- },
+ display: false,
},
scales: {
xAxes: [
@@ -179,6 +181,10 @@ export default function BarChart({
options.tooltips.custom = renderTooltip;
onUpdate(chart.current);
+
+ chart.current.update();
+
+ forceUpdate();
}
useEffect(() => {
@@ -202,23 +208,8 @@ export default function BarChart({
>
-
- {tooltip ? : null}
-
+
+
>
);
}
-
-const Tooltip = ({ title, value, label, labelColor }) => (
-
-);
diff --git a/components/metrics/BarChart.module.css b/components/metrics/BarChart.module.css
index cd26d3af..aea86a4c 100644
--- a/components/metrics/BarChart.module.css
+++ b/components/metrics/BarChart.module.css
@@ -1,43 +1,3 @@
.chart {
position: relative;
}
-
-.tooltip {
- color: var(--msgColor);
- pointer-events: none;
- z-index: 1;
-}
-
-.content {
- display: flex;
- flex-direction: column;
- justify-content: center;
- align-items: center;
- text-align: center;
-}
-
-.title {
- font-size: var(--font-size-xsmall);
- font-weight: 600;
-}
-
-.metric {
- display: flex;
- justify-content: center;
- align-items: center;
- font-size: var(--font-size-small);
- font-weight: 600;
-}
-
-.dot {
- position: relative;
- overflow: hidden;
- border-radius: 100%;
- margin-right: 8px;
- background: var(--gray50);
-}
-
-.color {
- width: 10px;
- height: 10px;
-}
diff --git a/components/metrics/ChartTooltip.js b/components/metrics/ChartTooltip.js
new file mode 100644
index 00000000..fb290b66
--- /dev/null
+++ b/components/metrics/ChartTooltip.js
@@ -0,0 +1,26 @@
+import React from 'react';
+import Dot from 'components/common/Dot';
+import styles from './ChartTooltip.module.css';
+import ReactTooltip from 'react-tooltip';
+
+export default function ChartTooltip({ chartId, tooltip }) {
+ if (!tooltip) {
+ return null;
+ }
+
+ const { title, value, label, labelColor } = tooltip;
+
+ return (
+
+
+
+
{title}
+
+
+ {value} {label}
+
+
+
+
+ );
+}
diff --git a/components/metrics/ChartTooltip.module.css b/components/metrics/ChartTooltip.module.css
new file mode 100644
index 00000000..cd26d3af
--- /dev/null
+++ b/components/metrics/ChartTooltip.module.css
@@ -0,0 +1,43 @@
+.chart {
+ position: relative;
+}
+
+.tooltip {
+ color: var(--msgColor);
+ pointer-events: none;
+ z-index: 1;
+}
+
+.content {
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ align-items: center;
+ text-align: center;
+}
+
+.title {
+ font-size: var(--font-size-xsmall);
+ font-weight: 600;
+}
+
+.metric {
+ display: flex;
+ justify-content: center;
+ align-items: center;
+ font-size: var(--font-size-small);
+ font-weight: 600;
+}
+
+.dot {
+ position: relative;
+ overflow: hidden;
+ border-radius: 100%;
+ margin-right: 8px;
+ background: var(--gray50);
+}
+
+.color {
+ width: 10px;
+ height: 10px;
+}
diff --git a/components/metrics/EventsChart.js b/components/metrics/EventsChart.js
index 9a5827b2..68556c96 100644
--- a/components/metrics/EventsChart.js
+++ b/components/metrics/EventsChart.js
@@ -16,7 +16,7 @@ export default function EventsChart({ websiteId, className, token }) {
const { query } = usePageQuery();
const shareToken = useShareToken();
- const { data } = useFetch(
+ const { data, loading } = useFetch(
`/api/website/${websiteId}/events`,
{
params: {
@@ -31,8 +31,10 @@ export default function EventsChart({ websiteId, className, token }) {
},
[modified],
);
+
const datasets = useMemo(() => {
if (!data) return [];
+ if (loading) return data;
const map = data.reduce((obj, { x, t, y }) => {
if (!obj[x]) {
@@ -59,15 +61,7 @@ export default function EventsChart({ websiteId, className, token }) {
borderWidth: 1,
};
});
- }, [data]);
-
- function handleCreate(options) {
- const legend = {
- position: 'bottom',
- };
-
- options.legend = legend;
- }
+ }, [data, loading]);
function handleUpdate(chart) {
chart.data.datasets = datasets;
@@ -85,9 +79,10 @@ export default function EventsChart({ websiteId, className, token }) {
className={className}
datasets={datasets}
unit={unit}
+ height={300}
records={getDateLength(startDate, endDate, unit)}
- onCreate={handleCreate}
onUpdate={handleUpdate}
+ loading={loading}
stacked
/>
);
diff --git a/components/metrics/Legend.js b/components/metrics/Legend.js
new file mode 100644
index 00000000..a40ff411
--- /dev/null
+++ b/components/metrics/Legend.js
@@ -0,0 +1,40 @@
+import React from 'react';
+import classNames from 'classnames';
+import Dot from 'components/common/Dot';
+import useLocale from 'hooks/useLocale';
+import styles from './Legend.module.css';
+import useForceUpdate from '../../hooks/useForceUpdate';
+
+export default function Legend({ chart }) {
+ const [locale] = useLocale();
+ const forceUpdate = useForceUpdate();
+
+ function handleClick(index) {
+ const meta = chart.getDatasetMeta(index);
+
+ meta.hidden = meta.hidden === null ? !chart.data.datasets[index].hidden : null;
+
+ chart.update();
+
+ forceUpdate();
+ }
+
+ if (!chart?.legend?.legendItems.find(({ text }) => text)) {
+ return null;
+ }
+
+ return (
+
+ {chart.legend.legendItems.map(({ text, fillStyle, datasetIndex, hidden }) => (
+
handleClick(datasetIndex)}
+ >
+
+ {text}
+
+ ))}
+
+ );
+}
diff --git a/components/metrics/Legend.module.css b/components/metrics/Legend.module.css
new file mode 100644
index 00000000..faa197e3
--- /dev/null
+++ b/components/metrics/Legend.module.css
@@ -0,0 +1,21 @@
+.legend {
+ display: flex;
+ justify-content: center;
+ flex-wrap: wrap;
+ margin-top: 10px;
+}
+
+.label {
+ display: flex;
+ align-items: center;
+ font-size: var(--font-size-xsmall);
+ cursor: pointer;
+}
+
+.label + .label {
+ margin-left: 20px;
+}
+
+.hidden {
+ color: var(--gray400);
+}
diff --git a/components/metrics/MetricsBar.js b/components/metrics/MetricsBar.js
index 33b6eaad..886ee5f0 100644
--- a/components/metrics/MetricsBar.js
+++ b/components/metrics/MetricsBar.js
@@ -31,7 +31,7 @@ export default function MetricsBar({ websiteId, className }) {
},
headers: { [TOKEN_HEADER]: shareToken?.token },
},
- [modified],
+ [url, modified],
);
const formatFunc = format ? formatLongNumber : formatNumber;
diff --git a/components/metrics/MetricsTable.js b/components/metrics/MetricsTable.js
index 16f61836..25bb4a08 100644
--- a/components/metrics/MetricsTable.js
+++ b/components/metrics/MetricsTable.js
@@ -1,5 +1,6 @@
import React, { useMemo } from 'react';
import { FormattedMessage } from 'react-intl';
+import firstBy from 'thenby';
import classNames from 'classnames';
import Link from 'components/common/Link';
import Loading from 'components/common/Loading';
@@ -55,9 +56,9 @@ export default function MetricsTable({
if (data) {
const items = percentFilter(dataFilter ? dataFilter(data, filterOptions) : data);
if (limit) {
- return items.filter((e, i) => i < limit);
+ return items.filter((e, i) => i < limit).sort(firstBy('y', -1).thenBy('x'));
}
- return items;
+ return items.sort(firstBy('y', -1).thenBy('x'));
}
return [];
}, [data, error, dataFilter, filterOptions]);
diff --git a/components/metrics/PageviewsChart.js b/components/metrics/PageviewsChart.js
index 79fe4917..e9d30449 100644
--- a/components/metrics/PageviewsChart.js
+++ b/components/metrics/PageviewsChart.js
@@ -45,8 +45,6 @@ export default function PageviewsChart({
id: 'metrics.page-views',
defaultMessage: 'Page views',
});
-
- chart.update();
};
if (!data) {
diff --git a/components/metrics/RealtimeHeader.module.css b/components/metrics/RealtimeHeader.module.css
index 8f948eb1..28aabc3e 100644
--- a/components/metrics/RealtimeHeader.module.css
+++ b/components/metrics/RealtimeHeader.module.css
@@ -1,3 +1,4 @@
.metrics {
display: flex;
+ margin-bottom: 10px;
}
diff --git a/components/metrics/RealtimeLog.js b/components/metrics/RealtimeLog.js
index c2de333b..18f064e8 100644
--- a/components/metrics/RealtimeLog.js
+++ b/components/metrics/RealtimeLog.js
@@ -1,5 +1,5 @@
import React, { useMemo, useState } from 'react';
-import { FormattedMessage } from 'react-intl';
+import { FormattedMessage, useIntl } from 'react-intl';
import { FixedSizeList } from 'react-window';
import firstBy from 'thenby';
import { format } from 'date-fns';
@@ -7,6 +7,7 @@ import Icon from 'components/common/Icon';
import Tag from 'components/common/Tag';
import Dot from 'components/common/Dot';
import FilterButtons from 'components/common/FilterButtons';
+import { devices } from 'components/messages';
import useLocale from 'hooks/useLocale';
import useCountryNames from 'hooks/useCountryNames';
import { BROWSERS } from 'lib/constants';
@@ -15,6 +16,7 @@ import Visitor from 'assets/visitor.svg';
import Eye from 'assets/eye.svg';
import { stringToColor } from 'lib/format';
import styles from './RealtimeLog.module.css';
+import NoData from '../common/NoData';
const TYPE_ALL = 0;
const TYPE_PAGEVIEW = 1;
@@ -28,6 +30,7 @@ const TYPE_ICONS = {
};
export default function RealtimeLog({ data, websites }) {
+ const intl = useIntl();
const [locale] = useLocale();
const countryNames = useCountryNames(locale);
const [filter, setFilter] = useState(TYPE_ALL);
@@ -116,9 +119,9 @@ export default function RealtimeLog({ data, websites }) {
defaultMessage="Visitor from {country} using {browser} on {os} {device}"
values={{
country: {countryNames[country]},
- browser: BROWSERS[browser],
- os,
- device,
+ browser: {BROWSERS[browser]},
+ os: {os},
+ device: {intl.formatMessage(devices[device])?.toLowerCase()},
}}
/>
);
@@ -159,6 +162,7 @@ export default function RealtimeLog({ data, websites }) {
+ {logs?.length === 0 &&
}
{Row}
diff --git a/components/metrics/WebsiteChart.js b/components/metrics/WebsiteChart.js
index 07ba5161..0f4b48c3 100644
--- a/components/metrics/WebsiteChart.js
+++ b/components/metrics/WebsiteChart.js
@@ -20,6 +20,7 @@ import { TOKEN_HEADER } from '../../lib/constants';
export default function WebsiteChart({
websiteId,
title,
+ domain,
stickyHeader = false,
showLink = false,
onDataLoad = () => {},
@@ -47,7 +48,7 @@ export default function WebsiteChart({
onDataLoad,
headers: { [TOKEN_HEADER]: shareToken?.token },
},
- [modified],
+ [url, modified],
);
const chartData = useMemo(() => {
@@ -66,7 +67,7 @@ export default function WebsiteChart({
return (
-
+
- {title}
+
+
+ {title}
+
diff --git a/components/pages/TestConsole.js b/components/pages/TestConsole.js
index fef6c620..b715b4a8 100644
--- a/components/pages/TestConsole.js
+++ b/components/pages/TestConsole.js
@@ -82,7 +82,12 @@ export default function TestConsole() {
diff --git a/components/pages/WebsiteDetails.js b/components/pages/WebsiteDetails.js
index 6c310909..81cde3ad 100644
--- a/components/pages/WebsiteDetails.js
+++ b/components/pages/WebsiteDetails.js
@@ -120,6 +120,7 @@ export default function WebsiteDetails({ websiteId }) {
- {data.map(({ website_id, name }) => (
+ {data.map(({ website_id, name, domain }) => (
-
+
))}
{data.length === 0 && (
diff --git a/components/settings/WebsiteSettings.js b/components/settings/WebsiteSettings.js
index 0ff6246d..686605f2 100644
--- a/components/settings/WebsiteSettings.js
+++ b/components/settings/WebsiteSettings.js
@@ -13,6 +13,7 @@ import ShareUrlForm from 'components/forms/ShareUrlForm';
import EmptyPlaceholder from 'components/common/EmptyPlaceholder';
import ButtonLayout from 'components/layout/ButtonLayout';
import Toast from 'components/common/Toast';
+import Favicon from 'components/common/Favicon';
import Pen from 'assets/pen.svg';
import Trash from 'assets/trash.svg';
import Plus from 'assets/plus.svg';
@@ -60,8 +61,9 @@ export default function WebsiteSettings() {
);
- const DetailsLink = ({ website_id, name }) => (
+ const DetailsLink = ({ website_id, name, domain }) => (
+
{name}
);
diff --git a/scripts/lang-ignore.json b/lang-ignore.json
similarity index 100%
rename from scripts/lang-ignore.json
rename to lang-ignore.json
diff --git a/lang/zh-CN.json b/lang/zh-CN.json
index 22b1be5b..101cd6cd 100644
--- a/lang/zh-CN.json
+++ b/lang/zh-CN.json
@@ -1,99 +1,99 @@
{
- "label.accounts": "账户",
- "label.add-account": "添加账户",
- "label.add-website": "添加网站",
- "label.administrator": "管理员",
- "label.all": "所有",
- "label.all-websites": "全部网站",
- "label.back": "返回",
- "label.cancel": "取消",
- "label.change-password": "更新密码",
- "label.confirm-password": "确认密码",
- "label.copy-to-clipboard": "复制",
- "label.current-password": "目前密码",
- "label.custom-range": "自定义时间段",
- "label.dashboard": "仪表板",
- "label.date-range": "多日",
- "label.default-date-range": "默认日期范围",
- "label.delete": "删除",
- "label.delete-account": "删除账户",
- "label.delete-website": "删除网站",
- "label.dismiss": "关闭",
- "label.domain": "域名",
- "label.edit": "编辑",
- "label.edit-account": "编辑账户",
- "label.edit-website": "编辑网站",
- "label.enable-share-url": "激活共享链接",
- "label.invalid": "输入无效",
- "label.invalid-domain": "无效域名",
- "label.last-days": "最近 {x} 天",
- "label.last-hours": "最近 {x} 小时",
- "label.logged-in-as": "登录名: {username}",
- "label.login": "登录",
- "label.logout": "退出",
- "label.more": "更多",
- "label.name": "名字",
- "label.new-password": "新密码",
- "label.password": "密码",
- "label.passwords-dont-match": "密码不一致",
- "label.profile": "个人资料",
- "label.realtime": "实时",
- "label.realtime-logs": "实时日志",
- "label.refresh": "刷新",
- "label.required": "必填",
- "label.reset": "重置",
- "label.save": "保存",
- "label.settings": "设置",
- "label.share-url": "共享链接",
- "label.single-day": "单日",
- "label.this-month": "本月",
- "label.this-week": "本周",
- "label.this-year": "今年",
- "label.timezone": "时区",
- "label.today": "今天",
- "label.tracking-code": "跟踪代码",
- "label.unknown": "未知",
- "label.username": "用户名",
- "label.view-details": "查看更多",
- "label.websites": "网站",
- "message.active-users": "当前在线 {x} 人",
- "message.confirm-delete": "你确定要删除{target}吗?",
- "message.copied": "复制成功!",
- "message.delete-warning": "所有相关数据将会被删除.",
- "message.failure": "出现错误.",
- "message.get-share-url": "获得共享链接",
- "message.get-tracking-code": "获得跟踪代码",
- "message.go-to-settings": "去设置",
- "message.incorrect-username-password": "用户名密码不正确.",
- "message.log.visitor": "来自 {country} 的访客在搭载 {os} 的 {device} 上使用 {browser} 进行访问.",
- "message.new-version-available": "umami 有新版本 {version} 发布啦!",
- "message.no-data-available": "无可用数据.",
- "message.no-websites-configured": "你还没有设置任何网站.",
- "message.page-not-found": "网页未找到.",
- "message.powered-by": "运行 {name}",
- "message.save-success": "成功保存.",
- "message.share-url": "这是 {target} 的共享链接.",
- "message.track-stats": "把以下代码放到你的网站的{head}部分来收集{target}的数据.",
- "message.type-delete": "在下面空格输入{delete}确认",
- "metrics.actions": "用户行为",
- "metrics.average-visit-time": "平均访问时间",
- "metrics.bounce-rate": "跳出率",
- "metrics.browsers": "浏览器",
- "metrics.countries": "国家",
- "metrics.device.desktop": "台式机",
- "metrics.device.laptop": "笔记本",
- "metrics.device.mobile": "手机",
- "metrics.device.tablet": "平板",
- "metrics.devices": "设备",
- "metrics.events": "行为类别",
- "metrics.filter.combined": "总和",
- "metrics.filter.domain-only": "只看域名",
- "metrics.filter.raw": "原始",
- "metrics.operating-systems": "操作系统",
- "metrics.page-views": "页面流量",
- "metrics.pages": "网页",
- "metrics.referrers": "指入域名",
- "metrics.unique-visitors": "独立访客",
- "metrics.views": "页面流量",
- "metrics.visitors": "独立访客"
-}
\ No newline at end of file
+ "label.accounts": "账户",
+ "label.add-account": "添加账户",
+ "label.add-website": "添加网站",
+ "label.administrator": "管理员",
+ "label.all": "所有",
+ "label.all-websites": "全部网站",
+ "label.back": "返回",
+ "label.cancel": "取消",
+ "label.change-password": "更新密码",
+ "label.confirm-password": "确认密码",
+ "label.copy-to-clipboard": "复制",
+ "label.current-password": "目前密码",
+ "label.custom-range": "自定义时间段",
+ "label.dashboard": "仪表板",
+ "label.date-range": "多日",
+ "label.default-date-range": "默认日期范围",
+ "label.delete": "删除",
+ "label.delete-account": "删除账户",
+ "label.delete-website": "删除网站",
+ "label.dismiss": "关闭",
+ "label.domain": "域名",
+ "label.edit": "编辑",
+ "label.edit-account": "编辑账户",
+ "label.edit-website": "编辑网站",
+ "label.enable-share-url": "激活共享链接",
+ "label.invalid": "输入无效",
+ "label.invalid-domain": "无效域名",
+ "label.last-days": "最近 {x} 天",
+ "label.last-hours": "最近 {x} 小时",
+ "label.logged-in-as": "登录名: {username}",
+ "label.login": "登录",
+ "label.logout": "退出",
+ "label.more": "更多",
+ "label.name": "名字",
+ "label.new-password": "新密码",
+ "label.password": "密码",
+ "label.passwords-dont-match": "密码不一致",
+ "label.profile": "个人资料",
+ "label.realtime": "实时",
+ "label.realtime-logs": "实时日志",
+ "label.refresh": "刷新",
+ "label.required": "必填",
+ "label.reset": "重置",
+ "label.save": "保存",
+ "label.settings": "设置",
+ "label.share-url": "共享链接",
+ "label.single-day": "单日",
+ "label.this-month": "本月",
+ "label.this-week": "本周",
+ "label.this-year": "今年",
+ "label.timezone": "时区",
+ "label.today": "今天",
+ "label.tracking-code": "跟踪代码",
+ "label.unknown": "未知",
+ "label.username": "用户名",
+ "label.view-details": "查看更多",
+ "label.websites": "网站",
+ "message.active-users": "当前在线 {x} 人",
+ "message.confirm-delete": "你确定要删除{target}吗?",
+ "message.copied": "复制成功!",
+ "message.delete-warning": "所有相关数据将会被删除.",
+ "message.failure": "出现错误.",
+ "message.get-share-url": "获得共享链接",
+ "message.get-tracking-code": "获得跟踪代码",
+ "message.go-to-settings": "去设置",
+ "message.incorrect-username-password": "用户名密码不正确.",
+ "message.log.visitor": "来自 {country} 的访客在搭载 {os} 的 {device} 上使用 {browser} 进行访问.",
+ "message.new-version-available": "umami 有新版本 {version} 发布啦!",
+ "message.no-data-available": "无可用数据.",
+ "message.no-websites-configured": "你还没有设置任何网站.",
+ "message.page-not-found": "网页未找到.",
+ "message.powered-by": "运行 {name}",
+ "message.save-success": "成功保存.",
+ "message.share-url": "这是 {target} 的共享链接.",
+ "message.track-stats": "把以下代码放到你的网站的{head}部分来收集{target}的数据.",
+ "message.type-delete": "在下面空格输入{delete}确认",
+ "metrics.actions": "用户行为",
+ "metrics.average-visit-time": "平均访问时间",
+ "metrics.bounce-rate": "跳出率",
+ "metrics.browsers": "浏览器",
+ "metrics.countries": "国家",
+ "metrics.device.desktop": "台式机",
+ "metrics.device.laptop": "笔记本",
+ "metrics.device.mobile": "手机",
+ "metrics.device.tablet": "平板",
+ "metrics.devices": "设备",
+ "metrics.events": "行为类别",
+ "metrics.filter.combined": "总和",
+ "metrics.filter.domain-only": "只看域名",
+ "metrics.filter.raw": "原始",
+ "metrics.operating-systems": "操作系统",
+ "metrics.page-views": "页面流量",
+ "metrics.pages": "网页",
+ "metrics.referrers": "指入域名",
+ "metrics.unique-visitors": "独立访客",
+ "metrics.views": "页面流量",
+ "metrics.visitors": "独立访客"
+}
diff --git a/lib/filters.js b/lib/filters.js
index 7fd5d833..d4853618 100644
--- a/lib/filters.js
+++ b/lib/filters.js
@@ -1,4 +1,3 @@
-import firstBy from 'thenby';
import { BROWSERS } from './constants';
import { removeTrailingSlash, removeWWW, getDomainName } from './url';
@@ -43,9 +42,7 @@ export const urlFilter = (data, { raw }) => {
return obj;
}, {});
- return Object.keys(map)
- .map(key => ({ x: key, y: map[key] }))
- .sort(firstBy('y', -1).thenBy('x'));
+ return Object.keys(map).map(key => ({ x: key, y: map[key] }));
};
export const refFilter = (data, { domain, domainOnly, raw }) => {
@@ -70,7 +67,7 @@ export const refFilter = (data, { domain, domainOnly, raw }) => {
}
if (domainOnly && hostname) {
- return hostname;
+ return removeWWW(hostname);
}
if (!origin || origin === 'null') {
@@ -113,9 +110,7 @@ export const refFilter = (data, { domain, domainOnly, raw }) => {
return obj;
}, {});
- return Object.keys(map)
- .map(key => ({ x: key, y: map[key], w: links[key] }))
- .sort(firstBy('y', -1).thenBy('x'));
+ return Object.keys(map).map(key => ({ x: key, y: map[key], w: links[key] }));
};
export const browserFilter = data =>
diff --git a/package.json b/package.json
index 3e7d6f64..b6dea4b1 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "umami",
- "version": "0.96.0",
+ "version": "1.0.0",
"description": "A simple, fast, website analytics alternative to Google Analytics. ",
"author": "Mike Cao ",
"license": "MIT",
diff --git a/scripts/check-lang.js b/scripts/check-lang.js
index a8c54538..b843efeb 100644
--- a/scripts/check-lang.js
+++ b/scripts/check-lang.js
@@ -2,7 +2,7 @@ const fs = require('fs');
const path = require('path');
const chalk = require('chalk');
const messages = require('../lang/en-US.json');
-const ignore = require('./lang-ignore.json');
+const ignore = require('../lang-ignore.json');
const dir = path.resolve(__dirname, '../lang');
const files = fs.readdirSync(dir);
@@ -13,7 +13,8 @@ files.forEach(file => {
const lang = require(`../lang/${file}`);
const id = file.replace('.json', '');
- console.log(chalk.yellowBright(`\n## ${file}`));
+ console.log(chalk.yellowBright(`\n## ${file.replace('.json', '')}`));
+ let count = 0;
keys.forEach(key => {
const orig = messages[key];
const check = lang[key];
@@ -21,7 +22,12 @@ files.forEach(file => {
if (!ignored && (!check || check === orig)) {
console.log(chalk.redBright('*'), chalk.greenBright(`${key}:`), orig);
+ count++;
}
});
+
+ if (count === 0) {
+ console.log('**👍 Complete!**');
+ }
}
});