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 }) => ( -
-
-
{title}
-
-
-
-
- {value} {label} -
-
-
-); 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 }) => ( + + ))} +
+ ); +} 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() {
- + Events
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!**'); + } } });