mirror of
https://github.com/umami-software/umami.git
synced 2026-02-15 10:05:36 +01:00
Fixed field parameters.
This commit is contained in:
parent
0f6cdf8b80
commit
7d2c361725
12 changed files with 90 additions and 88 deletions
|
|
@ -78,7 +78,7 @@
|
||||||
"@react-spring/web": "^9.7.5",
|
"@react-spring/web": "^9.7.5",
|
||||||
"@tanstack/react-query": "^5.68.0",
|
"@tanstack/react-query": "^5.68.0",
|
||||||
"@umami/prisma-client": "^0.16.0",
|
"@umami/prisma-client": "^0.16.0",
|
||||||
"@umami/react-zen": "^0.71.0",
|
"@umami/react-zen": "^0.73.0",
|
||||||
"@umami/redis-client": "^0.27.0",
|
"@umami/redis-client": "^0.27.0",
|
||||||
"bcryptjs": "^2.4.3",
|
"bcryptjs": "^2.4.3",
|
||||||
"chalk": "^4.1.2",
|
"chalk": "^4.1.2",
|
||||||
|
|
|
||||||
10
pnpm-lock.yaml
generated
10
pnpm-lock.yaml
generated
|
|
@ -45,8 +45,8 @@ importers:
|
||||||
specifier: ^0.16.0
|
specifier: ^0.16.0
|
||||||
version: 0.16.0(@prisma/client@6.5.0(prisma@6.5.0(typescript@5.8.2))(typescript@5.8.2))(@prisma/extension-read-replicas@0.4.0(@prisma/client@6.5.0(prisma@6.5.0(typescript@5.8.2))(typescript@5.8.2)))
|
version: 0.16.0(@prisma/client@6.5.0(prisma@6.5.0(typescript@5.8.2))(typescript@5.8.2))(@prisma/extension-read-replicas@0.4.0(@prisma/client@6.5.0(prisma@6.5.0(typescript@5.8.2))(typescript@5.8.2)))
|
||||||
'@umami/react-zen':
|
'@umami/react-zen':
|
||||||
specifier: ^0.71.0
|
specifier: ^0.73.0
|
||||||
version: 0.71.0(@babel/core@7.26.9)(@types/react@19.0.10)(immer@9.0.21)(use-sync-external-store@1.4.0(react@19.0.0))
|
version: 0.73.0(@babel/core@7.26.9)(@types/react@19.0.10)(immer@9.0.21)(use-sync-external-store@1.4.0(react@19.0.0))
|
||||||
'@umami/redis-client':
|
'@umami/redis-client':
|
||||||
specifier: ^0.27.0
|
specifier: ^0.27.0
|
||||||
version: 0.27.0
|
version: 0.27.0
|
||||||
|
|
@ -2964,8 +2964,8 @@ packages:
|
||||||
'@prisma/client': ^4.8.0
|
'@prisma/client': ^4.8.0
|
||||||
'@prisma/extension-read-replicas': ^0.3.0
|
'@prisma/extension-read-replicas': ^0.3.0
|
||||||
|
|
||||||
'@umami/react-zen@0.71.0':
|
'@umami/react-zen@0.73.0':
|
||||||
resolution: {integrity: sha512-SIfbDv/uW7mN5uebDeBeD+MnRiOrjsa2pJmVEdobyiM3ImdshpVVYxyBAJqAbiKbLF4OGZgRVbFsRfUcP/2Y0w==}
|
resolution: {integrity: sha512-cNagvTuo0QDaQPQzrfTBNafNnd1B5ygAMZyq7pi+O2U6Sr2pPi8iUW89VU07L2jjc1qoOncs66Ek08bsRHVHbQ==}
|
||||||
|
|
||||||
'@umami/redis-client@0.27.0':
|
'@umami/redis-client@0.27.0':
|
||||||
resolution: {integrity: sha512-SbHTpxhgeZyTBUSp2zdZM+XUtpsaSL4Tad8QXIEhEtjWhvvfoornyT5kLuyYCVtzSAT4daALeGmOO1z6EE1KcA==}
|
resolution: {integrity: sha512-SbHTpxhgeZyTBUSp2zdZM+XUtpsaSL4Tad8QXIEhEtjWhvvfoornyT5kLuyYCVtzSAT4daALeGmOO1z6EE1KcA==}
|
||||||
|
|
@ -10605,7 +10605,7 @@ snapshots:
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
|
||||||
'@umami/react-zen@0.71.0(@babel/core@7.26.9)(@types/react@19.0.10)(immer@9.0.21)(use-sync-external-store@1.4.0(react@19.0.0))':
|
'@umami/react-zen@0.73.0(@babel/core@7.26.9)(@types/react@19.0.10)(immer@9.0.21)(use-sync-external-store@1.4.0(react@19.0.0))':
|
||||||
dependencies:
|
dependencies:
|
||||||
'@fontsource/jetbrains-mono': 5.2.5
|
'@fontsource/jetbrains-mono': 5.2.5
|
||||||
'@react-aria/focus': 3.20.1(react-dom@19.0.0(react@19.0.0))(react@19.0.0)
|
'@react-aria/focus': 3.20.1(react-dom@19.0.0(react@19.0.0))(react@19.0.0)
|
||||||
|
|
|
||||||
|
|
@ -45,13 +45,13 @@ export function Nav(props: any) {
|
||||||
<SideNavSection>
|
<SideNavSection>
|
||||||
{links.map(({ href, label, icon }) => {
|
{links.map(({ href, label, icon }) => {
|
||||||
return (
|
return (
|
||||||
<Link key={href} href={href}>
|
<Link key={href} href={href} role="button">
|
||||||
<SideNavItem label={label} icon={icon} isSelected={pathname.startsWith(href)} />
|
<SideNavItem label={label} icon={icon} isSelected={pathname.startsWith(href)} />
|
||||||
</Link>
|
</Link>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</SideNavSection>
|
</SideNavSection>
|
||||||
<SideNavSection alignSelf="end"></SideNavSection>
|
<SideNavSection alignSelf="end">{``}</SideNavSection>
|
||||||
</SideNav>
|
</SideNav>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -11,9 +11,10 @@ export function FieldParameters() {
|
||||||
const { fields } = parameters || {};
|
const { fields } = parameters || {};
|
||||||
const { fields: fieldOptions } = useFields();
|
const { fields: fieldOptions } = useFields();
|
||||||
|
|
||||||
const handleAdd = (value: { name: any }) => {
|
const handleAdd = (value: string) => {
|
||||||
if (!fields.find(({ name }) => name === value.name)) {
|
if (!fields.find(({ name }) => name === value)) {
|
||||||
updateReport({ parameters: { fields: fields.concat(value) } });
|
const field = fieldOptions.find(({ name }) => name === value);
|
||||||
|
updateReport({ parameters: { fields: fields.concat(field) } });
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -21,30 +22,24 @@ export function FieldParameters() {
|
||||||
updateReport({ parameters: { fields: fields.filter(f => f.name !== name) } });
|
updateReport({ parameters: { fields: fields.filter(f => f.name !== name) } });
|
||||||
};
|
};
|
||||||
|
|
||||||
const AddButton = () => {
|
|
||||||
return (
|
|
||||||
<MenuTrigger>
|
|
||||||
<Button variant="quiet">
|
|
||||||
<Icon size="sm">
|
|
||||||
<Icons.Plus />
|
|
||||||
</Icon>
|
|
||||||
</Button>
|
|
||||||
<Popover placement="right top">
|
|
||||||
<FieldSelectForm
|
|
||||||
fields={fieldOptions.filter(({ name }) => !fields.find(f => f.name === name))}
|
|
||||||
onSelect={handleAdd}
|
|
||||||
showType={false}
|
|
||||||
/>
|
|
||||||
</Popover>
|
|
||||||
</MenuTrigger>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column gap="3">
|
<Column gap="3">
|
||||||
<Row justifyContent="space-between">
|
<Row justifyContent="space-between">
|
||||||
<Label>{formatMessage(labels.fields)}</Label>
|
<Label>{formatMessage(labels.fields)}</Label>
|
||||||
<AddButton />
|
<MenuTrigger>
|
||||||
|
<Button variant="quiet">
|
||||||
|
<Icon size="sm">
|
||||||
|
<Icons.Plus />
|
||||||
|
</Icon>
|
||||||
|
</Button>
|
||||||
|
<Popover placement="right top">
|
||||||
|
<FieldSelectForm
|
||||||
|
fields={fieldOptions.filter(({ name }) => !fields.find(f => f.name === name))}
|
||||||
|
onSelect={handleAdd}
|
||||||
|
showType={false}
|
||||||
|
/>
|
||||||
|
</Popover>
|
||||||
|
</MenuTrigger>
|
||||||
</Row>
|
</Row>
|
||||||
<ParameterList>
|
<ParameterList>
|
||||||
{fields.map(({ name }) => {
|
{fields.map(({ name }) => {
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,9 @@
|
||||||
import { Menu, MenuItem, Text, MenuSection } from '@umami/react-zen';
|
import { Menu, MenuItem, Text, MenuSection, Row } from '@umami/react-zen';
|
||||||
import { useMessages } from '@/components/hooks';
|
import { useMessages } from '@/components/hooks';
|
||||||
import { Key } from 'react';
|
|
||||||
|
|
||||||
export interface FieldSelectFormProps {
|
export interface FieldSelectFormProps {
|
||||||
fields?: any[];
|
fields?: any[];
|
||||||
onSelect?: (key: any) => void;
|
onSelect?: (value: any) => void;
|
||||||
showType?: boolean;
|
showType?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -12,13 +11,15 @@ export function FieldSelectForm({ fields = [], onSelect, showType = true }: Fiel
|
||||||
const { formatMessage, labels } = useMessages();
|
const { formatMessage, labels } = useMessages();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Menu onSelectionChange={key => onSelect(fields[key as any])}>
|
<Menu>
|
||||||
<MenuSection title={formatMessage(labels.fields)}>
|
<MenuSection title={formatMessage(labels.fields)}>
|
||||||
{fields.map(({ name, label, type }: any, index: Key) => {
|
{fields.map(({ name, label, type }) => {
|
||||||
return (
|
return (
|
||||||
<MenuItem key={index}>
|
<MenuItem key={name} id={name} onAction={() => onSelect(name)}>
|
||||||
<Text>{label || name}</Text>
|
<Row alignItems="center" justifyContent="space-between">
|
||||||
{showType && type && <Text color="muted">{type}</Text>}
|
<Text>{label || name}</Text>
|
||||||
|
{showType && type && <Text color="muted">{type}</Text>}
|
||||||
|
</Row>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
|
|
||||||
|
|
@ -8,8 +8,8 @@ import {
|
||||||
Icon,
|
Icon,
|
||||||
Popover,
|
Popover,
|
||||||
MenuTrigger,
|
MenuTrigger,
|
||||||
Focusable,
|
|
||||||
Text,
|
Text,
|
||||||
|
Pressable,
|
||||||
} from '@umami/react-zen';
|
} from '@umami/react-zen';
|
||||||
import { FilterSelectForm } from '../[reportId]/FilterSelectForm';
|
import { FilterSelectForm } from '../[reportId]/FilterSelectForm';
|
||||||
import { ParameterList } from '../[reportId]/ParameterList';
|
import { ParameterList } from '../[reportId]/ParameterList';
|
||||||
|
|
@ -48,32 +48,26 @@ export function FilterParameters() {
|
||||||
close();
|
close();
|
||||||
};
|
};
|
||||||
|
|
||||||
const AddButton = () => {
|
|
||||||
return (
|
|
||||||
<MenuTrigger>
|
|
||||||
<Button variant="quiet">
|
|
||||||
<Icon size="sm">
|
|
||||||
<Icons.Plus />
|
|
||||||
</Icon>
|
|
||||||
</Button>
|
|
||||||
<Popover placement="right top">
|
|
||||||
<FilterSelectForm
|
|
||||||
websiteId={websiteId}
|
|
||||||
fields={fields.filter(({ name }) => !filters.find(f => f.name === name))}
|
|
||||||
startDate={dateRange?.startDate}
|
|
||||||
endDate={dateRange?.endDate}
|
|
||||||
onChange={handleAdd}
|
|
||||||
/>
|
|
||||||
</Popover>
|
|
||||||
</MenuTrigger>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column gap="3">
|
<Column gap="3">
|
||||||
<Row justifyContent="space-between">
|
<Row justifyContent="space-between">
|
||||||
<Label>{formatMessage(labels.filters)}</Label>
|
<Label>{formatMessage(labels.filters)}</Label>
|
||||||
<AddButton />
|
<MenuTrigger>
|
||||||
|
<Button variant="quiet">
|
||||||
|
<Icon size="sm">
|
||||||
|
<Icons.Plus />
|
||||||
|
</Icon>
|
||||||
|
</Button>
|
||||||
|
<Popover placement="right top">
|
||||||
|
<FilterSelectForm
|
||||||
|
websiteId={websiteId}
|
||||||
|
fields={fields.filter(({ name }) => !filters.find(f => f.name === name))}
|
||||||
|
startDate={dateRange?.startDate}
|
||||||
|
endDate={dateRange?.endDate}
|
||||||
|
onChange={handleAdd}
|
||||||
|
/>
|
||||||
|
</Popover>
|
||||||
|
</MenuTrigger>
|
||||||
</Row>
|
</Row>
|
||||||
<ParameterList>
|
<ParameterList>
|
||||||
{filters.map(
|
{filters.map(
|
||||||
|
|
@ -117,15 +111,15 @@ const FilterParameter = ({
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MenuTrigger>
|
<MenuTrigger>
|
||||||
<Focusable>
|
<Pressable>
|
||||||
<Row gap="3" alignItems="center">
|
<Row role="button" gap="3" alignItems="center">
|
||||||
<Text>{label}</Text>
|
<Text>{label}</Text>
|
||||||
<Text size="2" transform="uppercase">
|
<Text size="2" transform="uppercase">
|
||||||
{operatorLabels[operator]}
|
{operatorLabels[operator]}
|
||||||
</Text>
|
</Text>
|
||||||
<Text weight="bold">{value}</Text>
|
<Text weight="bold">{value}</Text>
|
||||||
</Row>
|
</Row>
|
||||||
</Focusable>
|
</Pressable>
|
||||||
<Popover placement="right top">
|
<Popover placement="right top">
|
||||||
{(close: any) => (
|
{(close: any) => (
|
||||||
<FieldFilterEditForm
|
<FieldFilterEditForm
|
||||||
|
|
|
||||||
|
|
@ -12,7 +12,7 @@ export function ParameterList({ children }: ParameterListProps) {
|
||||||
const { formatMessage, labels } = useMessages();
|
const { formatMessage, labels } = useMessages();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column>
|
<Column gap="3">
|
||||||
{!children && <Empty message={formatMessage(labels.none)} />}
|
{!children && <Empty message={formatMessage(labels.none)} />}
|
||||||
{children}
|
{children}
|
||||||
</Column>
|
</Column>
|
||||||
|
|
@ -40,6 +40,7 @@ const Item = ({
|
||||||
border
|
border
|
||||||
borderRadius="2"
|
borderRadius="2"
|
||||||
paddingLeft="3"
|
paddingLeft="3"
|
||||||
|
shadow="2"
|
||||||
>
|
>
|
||||||
{icon && <Icon>{icon}</Icon>}
|
{icon && <Icon>{icon}</Icon>}
|
||||||
<Text>{children}</Text>
|
<Text>{children}</Text>
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { Button, Icon, Icons, Box, MenuTrigger, Popover, Text } from '@umami/react-zen';
|
import { Button, Icon, Icons, MenuTrigger, Popover, Text } from '@umami/react-zen';
|
||||||
import { FilterSelectForm } from '@/app/(main)/reports/[reportId]/FilterSelectForm';
|
import { FilterSelectForm } from '@/app/(main)/reports/[reportId]/FilterSelectForm';
|
||||||
import { useFields, useMessages, useNavigation, useDateRange } from '@/components/hooks';
|
import { useFields, useMessages, useNavigation, useDateRange } from '@/components/hooks';
|
||||||
import { OPERATOR_PREFIXES } from '@/lib/constants';
|
import { OPERATOR_PREFIXES } from '@/lib/constants';
|
||||||
|
|
@ -34,21 +34,19 @@ export function WebsiteFilterButton({
|
||||||
</Icon>
|
</Icon>
|
||||||
{showText && <Text>{formatMessage(labels.filter)}</Text>}
|
{showText && <Text>{formatMessage(labels.filter)}</Text>}
|
||||||
</Button>
|
</Button>
|
||||||
<Popover placement="bottom end">
|
<Popover placement="bottom start">
|
||||||
{({ close }: any) => {
|
{({ close }: any) => {
|
||||||
return (
|
return (
|
||||||
<Box padding="3" backgroundColor="1">
|
<FilterSelectForm
|
||||||
<FilterSelectForm
|
websiteId={websiteId}
|
||||||
websiteId={websiteId}
|
fields={fields}
|
||||||
fields={fields}
|
startDate={startDate}
|
||||||
startDate={startDate}
|
endDate={endDate}
|
||||||
endDate={endDate}
|
onChange={value => {
|
||||||
onChange={value => {
|
handleAddFilter(value);
|
||||||
handleAddFilter(value);
|
close();
|
||||||
close();
|
}}
|
||||||
}}
|
/>
|
||||||
/>
|
|
||||||
</Box>
|
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
</Popover>
|
</Popover>
|
||||||
|
|
|
||||||
|
|
@ -2,5 +2,15 @@ import { Box } from '@umami/react-zen';
|
||||||
import type { BoxProps } from '@umami/react-zen/Box';
|
import type { BoxProps } from '@umami/react-zen/Box';
|
||||||
|
|
||||||
export function Panel(props: BoxProps) {
|
export function Panel(props: BoxProps) {
|
||||||
return <Box padding="6" border borderRadius="3" backgroundColor="solid" shadow="4" {...props} />;
|
return (
|
||||||
|
<Box
|
||||||
|
padding="6"
|
||||||
|
border
|
||||||
|
borderRadius="3"
|
||||||
|
backgroundColor
|
||||||
|
shadow="4"
|
||||||
|
position="relative"
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
import { useState, Key } from 'react';
|
import { useState, Key, Fragment } from 'react';
|
||||||
import { Icon, Modal, Select, Text, Row, ListItem, ListSeparator, Dialog } from '@umami/react-zen';
|
import { Icon, Modal, Select, Text, Row, ListItem, ListSeparator, Dialog } from '@umami/react-zen';
|
||||||
import { endOfYear, isSameDay } from 'date-fns';
|
import { endOfYear, isSameDay } from 'date-fns';
|
||||||
import { DatePickerForm } from '@/components/metrics/DatePickerForm';
|
import { DatePickerForm } from '@/components/metrics/DatePickerForm';
|
||||||
|
|
@ -101,12 +101,12 @@ export function DateFilter({
|
||||||
>
|
>
|
||||||
{options.map(({ label, value, divider }: any) => {
|
{options.map(({ label, value, divider }: any) => {
|
||||||
return (
|
return (
|
||||||
<>
|
<Fragment key={label}>
|
||||||
{divider && <ListSeparator />}
|
{divider && <ListSeparator />}
|
||||||
<ListItem key={value} id={value}>
|
<ListItem key={label} id={value}>
|
||||||
{label}
|
{label}
|
||||||
</ListItem>
|
</ListItem>
|
||||||
</>
|
</Fragment>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</Select>
|
</Select>
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,5 @@
|
||||||
.card {
|
.card {
|
||||||
border-right: 1px solid var(--border-color);
|
border-right: 1px solid var(--border-color);
|
||||||
padding: 0 50px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.card:last-child {
|
.card:last-child {
|
||||||
|
|
|
||||||
|
|
@ -34,8 +34,12 @@ export const MetricCard = ({
|
||||||
const prevProps = useSpring({ x: Number(diff) || 0, from: { x: 0 } });
|
const prevProps = useSpring({ x: Number(diff) || 0, from: { x: 0 } });
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column className={styles.card} justifyContent="center">
|
<Column className={styles.card} justifyContent="center" paddingX="8">
|
||||||
{showLabel && <Text weight="bold">{label}</Text>}
|
{showLabel && (
|
||||||
|
<Text weight="bold" wrap="nowrap">
|
||||||
|
{label}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
<Text size="8" weight="bold" wrap="nowrap">
|
<Text size="8" weight="bold" wrap="nowrap">
|
||||||
<AnimatedDiv title={value?.toString()}>{props?.x?.to(x => formatValue(x))}</AnimatedDiv>
|
<AnimatedDiv title={value?.toString()}>{props?.x?.to(x => formatValue(x))}</AnimatedDiv>
|
||||||
</Text>
|
</Text>
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue