Fixed field parameters.

This commit is contained in:
Mike Cao 2025-03-27 15:33:24 -07:00
parent 0f6cdf8b80
commit 7d2c361725
12 changed files with 90 additions and 88 deletions

View file

@ -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
View file

@ -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)

View file

@ -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>
); );
} }

View file

@ -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 }) => {

View file

@ -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>
); );
})} })}

View file

@ -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

View file

@ -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>

View file

@ -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>

View file

@ -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}
/>
);
} }

View file

@ -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>

View file

@ -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 {

View file

@ -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>