Event data report UI.

This commit is contained in:
Mike Cao 2023-07-01 22:02:49 -07:00
parent 6316a0b917
commit 9d7862cbd6
36 changed files with 660 additions and 254 deletions

View File

@ -1,15 +1,15 @@
import classNames from 'classnames'; import classNames from 'classnames';
import styles from './NoData.module.css'; import styles from './Empty.module.css';
import useMessages from 'hooks/useMessages'; import useMessages from 'hooks/useMessages';
export function NoData({ className }) { export function Empty({ message, className }) {
const { formatMessage, messages } = useMessages(); const { formatMessage, messages } = useMessages();
return ( return (
<div className={classNames(styles.container, className)}> <div className={classNames(styles.container, className)}>
{formatMessage(messages.noDataAvailable)} {message || formatMessage(messages.noDataAvailable)}
</div> </div>
); );
} }
export default NoData; export default Empty;

View File

@ -129,7 +129,6 @@ export const labels = defineMessages({
urls: { id: 'label.urls', defaultMessage: 'URLs' }, urls: { id: 'label.urls', defaultMessage: 'URLs' },
add: { id: 'label.add', defaultMessage: 'Add' }, add: { id: 'label.add', defaultMessage: 'Add' },
window: { id: 'label.window', defaultMessage: 'Window' }, window: { id: 'label.window', defaultMessage: 'Window' },
addUrl: { id: 'label.add-url', defaultMessage: 'Add URL' },
runQuery: { id: 'label.run-query', defaultMessage: 'Run query' }, runQuery: { id: 'label.run-query', defaultMessage: 'Run query' },
field: { id: 'label.field', defaultMessage: 'Field' }, field: { id: 'label.field', defaultMessage: 'Field' },
fields: { id: 'label.fields', defaultMessage: 'Fields' }, fields: { id: 'label.fields', defaultMessage: 'Fields' },
@ -137,6 +136,20 @@ export const labels = defineMessages({
description: { id: 'labels.description', defaultMessage: 'Description' }, description: { id: 'labels.description', defaultMessage: 'Description' },
untitled: { id: 'labels.untitled', defaultMessage: 'Untitled' }, untitled: { id: 'labels.untitled', defaultMessage: 'Untitled' },
type: { id: 'labels.type', defaultMessage: 'Type' }, type: { id: 'labels.type', defaultMessage: 'Type' },
filters: { id: 'labels.filters', defaultMessage: 'Filters' },
groupBy: { id: 'labels.group-by', defaultMessage: 'Group by' },
true: { id: 'labels.true', defaultMessage: 'True' },
false: { id: 'labels.false', defaultMessage: 'False' },
equals: { id: 'labels.equals', defaultMessage: 'Equals' },
doesNotEqual: { id: 'labels.does-not-equal', defaultMessage: 'Does not equal' },
greaterThan: { id: 'labels.greater-than', defaultMessage: 'Greater than' },
lessThan: { id: 'labels.less-than', defaultMessage: 'Less than' },
greaterThanEquals: { id: 'labels.greater-than-equals', defaultMessage: 'Greater than or equals' },
lessThanEquals: { id: 'labels.less-than-equals', defaultMessage: 'Less than or equals' },
contains: { id: 'labels.contains', defaultMessage: 'Contains' },
doesNotContain: { id: 'labels.does-not-contain', defaultMessage: 'Does not contain' },
before: { id: 'labels.before', defaultMessage: 'Before' },
after: { id: 'labels.after', defaultMessage: 'After' },
}); });
export const messages = defineMessages({ export const messages = defineMessages({
@ -244,4 +257,8 @@ export const messages = defineMessages({
id: 'message.incorrect-username-password', id: 'message.incorrect-username-password',
defaultMessage: 'Incorrect username and/or password.', defaultMessage: 'Incorrect username and/or password.',
}, },
noEventData: {
id: 'message.no-event-data',
defaultMessage: 'No event data is available.',
},
}); });

View File

@ -3,7 +3,7 @@ import useMeasure from 'react-use-measure';
import { FixedSizeList } from 'react-window'; import { FixedSizeList } from 'react-window';
import { useSpring, animated, config } from 'react-spring'; import { useSpring, animated, config } from 'react-spring';
import classNames from 'classnames'; import classNames from 'classnames';
import NoData from 'components/common/NoData'; import Empty from 'components/common/Empty';
import { formatNumber, formatLongNumber } from 'lib/format'; import { formatNumber, formatLongNumber } from 'lib/format';
import useMessages from 'hooks/useMessages'; import useMessages from 'hooks/useMessages';
import styles from './DataTable.module.css'; import styles from './DataTable.module.css';
@ -55,7 +55,7 @@ export function DataTable({
</div> </div>
</div> </div>
<div ref={ref} className={styles.body}> <div ref={ref} className={styles.body}>
{data?.length === 0 && <NoData />} {data?.length === 0 && <Empty />}
{virtualize && data.length > 0 ? ( {virtualize && data.length > 0 ? (
<FixedSizeList height={bounds.height} itemCount={data.length} itemSize={30}> <FixedSizeList height={bounds.height} itemCount={data.length} itemSize={30}>
{Row} {Row}

View File

@ -3,7 +3,7 @@ import { StatusLight, Icon, Text } from 'react-basics';
import { FixedSizeList } from 'react-window'; import { FixedSizeList } from 'react-window';
import firstBy from 'thenby'; import firstBy from 'thenby';
import FilterButtons from 'components/common/FilterButtons'; import FilterButtons from 'components/common/FilterButtons';
import NoData from 'components/common/NoData'; import Empty from 'components/common/Empty';
import useLocale from 'hooks/useLocale'; import useLocale from 'hooks/useLocale';
import useCountryNames from 'hooks/useCountryNames'; import useCountryNames from 'hooks/useCountryNames';
import { BROWSERS } from 'lib/constants'; import { BROWSERS } from 'lib/constants';
@ -144,7 +144,7 @@ export function RealtimeLog({ data, websiteDomain }) {
<FilterButtons items={buttons} selectedKey={filter} onSelect={setFilter} /> <FilterButtons items={buttons} selectedKey={filter} onSelect={setFilter} />
<div className={styles.header}>{formatMessage(labels.activityLog)}</div> <div className={styles.header}>{formatMessage(labels.activityLog)}</div>
<div className={styles.body}> <div className={styles.body}>
{logs?.length === 0 && <NoData />} {logs?.length === 0 && <Empty />}
{logs?.length > 0 && ( {logs?.length > 0 && (
<FixedSizeList height={500} itemCount={logs.length} itemSize={50}> <FixedSizeList height={500} itemCount={logs.length} itemSize={50}>
{Row} {Row}

View File

@ -0,0 +1,38 @@
import { Form, FormRow, Menu, Item } from 'react-basics';
const options = {
number: [
{ label: 'SUM', value: 'sum' },
{ label: 'AVERAGE', value: 'average' },
{ label: 'MIN', value: 'min' },
{ label: 'MAX', value: 'max' },
],
date: [
{ label: 'MIN', value: 'min' },
{ label: 'MAX', value: 'max' },
],
string: [
{ label: 'COUNT', value: 'count' },
{ label: 'DISTINCT', value: 'distinct' },
],
};
export default function FieldAggregateForm({ name, type, onSelect }) {
const items = options[type];
const handleSelect = value => {
onSelect({ name, value });
};
return (
<Form>
<FormRow label={name}>
<Menu onSelect={handleSelect}>
{items.map(({ label, value }) => {
return <Item key={value}>{label}</Item>;
})}
</Menu>
</FormRow>
</Form>
);
}

View File

@ -0,0 +1,57 @@
import { useState } from 'react';
import { Form, FormRow, Menu, Item, Flexbox, Dropdown, TextField, Button } from 'react-basics';
import { useFilters } from 'hooks';
import styles from './FieldFilterForm.module.css';
export default function FieldFilterForm({ name, type, onSelect }) {
const [filter, setFilter] = useState('');
const [value, setValue] = useState('');
const { filters, types } = useFilters();
const items = types[type];
const renderValue = value => {
return filters[value];
};
if (type === 'boolean') {
return (
<Form>
<FormRow label={name}>
<Menu onSelect={value => onSelect({ name, value: ['eq', value] })}>
{items.map(value => {
return <Item key={value}>{filters[value]}</Item>;
})}
</Menu>
</FormRow>
</Form>
);
}
return (
<Form>
<FormRow label={name} className={styles.filter}>
<Flexbox gap={10}>
<Dropdown
className={styles.dropdown}
items={items}
value={filter}
renderValue={renderValue}
onChange={setFilter}
>
{value => {
return <Item key={value}>{filters[value]}</Item>;
}}
</Dropdown>
<TextField value={value} onChange={e => setValue(e.target.value)} autoFocus={true} />
</Flexbox>
<Button
variant="primary"
onClick={() => onSelect({ name, value: [filter, value] })}
disabled={!filter || !value}
>
Add
</Button>
</FormRow>
</Form>
);
}

View File

@ -0,0 +1,17 @@
.selected {
font-weight: bold;
}
.popup {
display: flex;
}
.filter {
display: flex;
flex-direction: column;
gap: 20px;
}
.dropdown {
min-width: 60px;
}

View File

@ -0,0 +1,24 @@
import { Menu, Item, Form, FormRow } from 'react-basics';
import { useMessages } from 'hooks';
import styles from './FieldSelectForm.module.css';
export default function FieldSelectForm({ fields, onSelect }) {
const { formatMessage, labels } = useMessages();
return (
<Form>
<FormRow label={formatMessage(labels.fields)}>
<Menu className={styles.menu} onSelect={key => onSelect(fields[key])}>
{fields.map(({ name, type }, index) => {
return (
<Item key={index} className={styles.item}>
<div>{name}</div>
<div className={styles.type}>{type}</div>
</Item>
);
})}
</Menu>
</FormRow>
</Form>
);
}

View File

@ -0,0 +1,20 @@
.menu {
width: 360px;
max-height: 300px;
overflow: auto;
}
.item {
display: flex;
flex-direction: row;
justify-content: space-between;
border-radius: var(--border-radius);
}
.item:hover {
background: var(--base75);
}
.type {
color: var(--font-color300);
}

View File

@ -0,0 +1,33 @@
import { Icon, Text, TooltipPopup } from 'react-basics';
import Icons from 'components/icons';
import Empty from 'components/common/Empty';
import { useMessages } from 'hooks';
import styles from './ParameterList.module.css';
export function ParameterList({ items = [], children, onRemove }) {
const { formatMessage, labels } = useMessages();
return (
<div className={styles.list}>
{!items.length && <Empty message={formatMessage(labels.none)} />}
{items.map((item, index) => {
return (
<div key={index} className={styles.item}>
<Text>{typeof children === 'function' ? children(item) : item}</Text>
<TooltipPopup
className={styles.icon}
label={formatMessage(labels.remove)}
position="right"
>
<Icon onClick={onRemove.bind(null, index)}>
<Icons.Close />
</Icon>
</TooltipPopup>
</div>
);
})}
</div>
);
}
export default ParameterList;

View File

@ -1,10 +1,10 @@
.urls { .list {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 16px; gap: 16px;
} }
.url { .item {
display: flex; display: flex;
flex-direction: row; flex-direction: row;
align-items: center; align-items: center;

View File

@ -0,0 +1,30 @@
import { createPortal } from 'react-dom';
import { useDocumentClick, useKeyDown } from 'react-basics';
import classNames from 'classnames';
import styles from './PopupForm.module.css';
export function PopupForm({ element, className, children, onClose }) {
const { right, top } = element.getBoundingClientRect();
const style = { position: 'absolute', left: right, top };
useKeyDown('Escape', onClose);
useDocumentClick(e => {
if (e.target !== element && !element?.parentElement?.contains(e.target)) {
onClose();
}
});
const handleClick = e => {
e.stopPropagation();
};
return createPortal(
<div className={classNames(styles.form, className)} style={style} onClick={handleClick}>
{children}
</div>,
document.body,
);
}
export default PopupForm;

View File

@ -0,0 +1,10 @@
.form {
position: absolute;
background: var(--base50);
min-width: 300px;
padding: 20px;
margin-left: 30px;
border: 1px solid var(--base400);
border-radius: var(--border-radius);
box-shadow: 0 0 0 5px rgba(0, 0, 0, 0.1);
}

View File

@ -8,6 +8,8 @@ export const ReportContext = createContext(null);
export function Report({ reportId, defaultParameters, children, ...props }) { export function Report({ reportId, defaultParameters, children, ...props }) {
const report = useReport(reportId, defaultParameters); const report = useReport(reportId, defaultParameters);
//console.log({ report });
return ( return (
<ReportContext.Provider value={{ ...report }}> <ReportContext.Provider value={{ ...report }}>
<Page {...props} className={styles.container}> <Page {...props} className={styles.container}>

View File

@ -0,0 +1,13 @@
import FunnelReport from './funnel/FunnelReport';
import EventDataReport from './event-data/EventDataReport';
const reports = {
funnel: FunnelReport,
'event-data': EventDataReport,
};
export default function ReportDetails({ reportId, reportType }) {
const Report = reports[reportType];
return <Report reportId={reportId} />;
}

View File

@ -1,17 +1,25 @@
import { useContext, useRef } from 'react'; import { useContext, useRef } from 'react';
import { useApi, useMessages } from 'hooks'; import { useApi, useMessages } from 'hooks';
import { Form, FormRow, FormButtons, SubmitButton, Loading } from 'react-basics'; import { Form, FormRow, FormButtons, SubmitButton, PopupTrigger, Icon, Popup } from 'react-basics';
import { ReportContext } from 'components/pages/reports/Report'; import { ReportContext } from 'components/pages/reports/Report';
import NoData from 'components/common/NoData'; import Empty from 'components/common/Empty';
import styles from './EventDataParameters.module.css';
import { DATA_TYPES } from 'lib/constants'; import { DATA_TYPES } from 'lib/constants';
import BaseParameters from '../BaseParameters'; import BaseParameters from '../BaseParameters';
import FieldAddForm from './FieldAddForm';
import ParameterList from '../ParameterList';
import Icons from 'components/icons';
import styles from './EventDataParameters.module.css';
function useFields(websiteId, startDate, endDate) { function useFields(websiteId, startDate, endDate) {
const { get, useQuery } = useApi(); const { get, useQuery } = useApi();
const { data, error, isLoading } = useQuery( const { data, error, isLoading } = useQuery(
['fields', websiteId, startDate, endDate], ['fields', websiteId, startDate, endDate],
() => get('/reports/event-data', { websiteId, startAt: +startDate, endAt: +endDate }), () =>
get('/reports/event-data', {
websiteId,
startAt: +startDate,
endAt: +endDate,
}),
{ enabled: !!(websiteId && startDate && endDate) }, { enabled: !!(websiteId && startDate && endDate) },
); );
@ -19,35 +27,111 @@ function useFields(websiteId, startDate, endDate) {
} }
export function EventDataParameters() { export function EventDataParameters() {
const { report, runReport, isRunning } = useContext(ReportContext); const { report, runReport, updateReport, isRunning } = useContext(ReportContext);
const { formatMessage, labels } = useMessages(); const { formatMessage, labels, messages } = useMessages();
const ref = useRef(null); const ref = useRef(null);
const { parameters } = report || {}; const { parameters } = report || {};
const { websiteId, dateRange } = parameters || {}; const { websiteId, dateRange, fields, filters, groups } = parameters || {};
const { startDate, endDate } = dateRange || {}; const { startDate, endDate } = dateRange || {};
const queryDisabled = !websiteId || !dateRange; const queryDisabled = !websiteId || !dateRange;
const { data, error, isLoading } = useFields(websiteId, startDate, endDate); const { data, error } = useFields(websiteId, startDate, endDate);
const parametersSelected = websiteId && startDate && endDate;
const hasData = data?.length !== 0;
const parameterGroups = [
{ label: formatMessage(labels.fields), type: 'fields' },
{ label: formatMessage(labels.filters), type: 'filters' },
{ label: formatMessage(labels.groupBy), type: 'groups' },
];
const parameterData = {
fields,
filters,
groups,
};
const handleSubmit = values => { const handleSubmit = values => {
runReport(values); runReport(values);
}; };
const handleAdd = (type, value) => {
const data = parameterData[type];
updateReport({ parameters: { [type]: data.concat(value) } });
};
const handleRemove = (type, index) => {
const data = [...parameterData[type]];
data.splice(index, 1);
updateReport({ parameters: { [type]: data } });
};
const AddButton = ({ type }) => {
return (
<PopupTrigger>
<Icon>
<Icons.Plus />
</Icon>
<Popup position="bottom" alignment="start">
{(close, element) => {
return (
<FieldAddForm
type={type}
fields={data.map(({ eventKey, eventDataType }) => ({
name: eventKey,
type: DATA_TYPES[eventDataType],
}))}
element={element}
onAdd={handleAdd}
onClose={close}
/>
);
}}
</Popup>
</PopupTrigger>
);
};
return ( return (
<Form ref={ref} values={parameters} error={error} onSubmit={handleSubmit}> <Form ref={ref} values={parameters} error={error} onSubmit={handleSubmit}>
<BaseParameters /> <BaseParameters />
<FormRow label={formatMessage(labels.fields)}> {!hasData && <Empty message={formatMessage(messages.noEventData)} />}
<div className={styles.fields}> {parametersSelected &&
{!data?.length && <NoData />} hasData &&
{data?.map?.(({ eventKey, eventDataType }) => { parameterGroups.map(({ label, type }) => {
return ( return (
<div className={styles.field} key={eventKey}> <FormRow key={label} label={label} action={<AddButton type={type} onAdd={handleAdd} />}>
<div className={styles.key}>{eventKey}</div> <ParameterList
<div className={styles.type}>{DATA_TYPES[eventDataType]}</div> items={parameterData[type]}
</div> onRemove={index => handleRemove(type, index)}
); >
})} {({ name, value }) => {
</div> return (
</FormRow> <div className={styles.parameter}>
{type === 'fields' && (
<>
<div className={styles.op}>{value}</div>
<div>{name}</div>
</>
)}
{type === 'filters' && (
<>
<div>{name}</div>
<div className={styles.op}>{value[0]}</div>
<div>{value[1]}</div>
</>
)}
{type === 'groups' && (
<>
<div>{name}</div>
</>
)}
</div>
);
}}
</ParameterList>
</FormRow>
);
})}
<FormButtons> <FormButtons>
<SubmitButton variant="primary" disabled={queryDisabled} loading={isRunning}> <SubmitButton variant="primary" disabled={queryDisabled} loading={isRunning}>
{formatMessage(labels.runQuery)} {formatMessage(labels.runQuery)}

View File

@ -1,27 +1,8 @@
.fields { .parameter {
max-height: 300px;
overflow: auto;
}
.field {
display: flex; display: flex;
flex-direction: row; gap: 10px;
justify-content: space-between;
align-items: center;
height: 30px;
cursor: pointer;
padding: 4px;
border-radius: var(--border-radius);
} }
.field:hover { .op {
background: var(--base75); font-weight: bold;
}
.key {
font-weight: 400;
}
.type {
color: var(--font-color300);
} }

View File

@ -4,10 +4,11 @@ import ReportMenu from '../ReportMenu';
import ReportBody from '../ReportBody'; import ReportBody from '../ReportBody';
import EventDataParameters from './EventDataParameters'; import EventDataParameters from './EventDataParameters';
import Nodes from 'assets/nodes.svg'; import Nodes from 'assets/nodes.svg';
import EventDataTable from './EventDataTable';
const defaultParameters = { const defaultParameters = {
type: 'event-data', type: 'event-data',
parameters: { fields: [], filters: [] }, parameters: { fields: [], filters: [], groups: [] },
}; };
export default function EventDataReport({ reportId }) { export default function EventDataReport({ reportId }) {
@ -17,7 +18,9 @@ export default function EventDataReport({ reportId }) {
<ReportMenu> <ReportMenu>
<EventDataParameters /> <EventDataParameters />
</ReportMenu> </ReportMenu>
<ReportBody>hi.</ReportBody> <ReportBody>
<EventDataTable />
</ReportBody>
</Report> </Report>
); );
} }

View File

@ -0,0 +1,20 @@
import { useContext } from 'react';
import DataTable from 'components/metrics/DataTable';
import { useMessages } from 'hooks';
import { ReportContext } from '../Report';
export function EventDataTable() {
const { report } = useContext(ReportContext);
const { formatMessage, labels } = useMessages();
return (
<DataTable
data={report?.data}
title={formatMessage(labels.eventData)}
metric="#"
showPercentage={true}
/>
);
}
export default EventDataTable;

View File

@ -1,27 +1,35 @@
import { useMessages } from 'hooks'; import { useState } from 'react';
import { Button, Form, FormButtons, FormRow } from 'react-basics'; import { createPortal } from 'react-dom';
import PopupForm from '../PopupForm';
import FieldSelectForm from '../FieldSelectForm';
import FieldAggregateForm from '../FieldAggregateForm';
import FieldFilterForm from '../FieldFilterForm';
import styles from './FieldAddForm.module.css';
export function FieldAddForm({ onClose }) { export function FieldAddForm({ fields = [], type, element, onAdd, onClose }) {
const { formatMessage, labels } = useMessages(); const [selected, setSelected] = useState();
const handleSave = () => { const handleSelect = value => {
if (type === 'groups') {
handleSave(value);
return;
}
setSelected(value);
};
const handleSave = value => {
onAdd(type, value);
onClose(); onClose();
}; };
const handleClose = () => { return createPortal(
onClose(); <PopupForm className={styles.popup} element={element} onClose={onClose}>
}; {!selected && <FieldSelectForm fields={fields} type={type} onSelect={handleSelect} />}
{selected && type === 'fields' && <FieldAggregateForm {...selected} onSelect={handleSave} />}
return ( {selected && type === 'filters' && <FieldFilterForm {...selected} onSelect={handleSave} />}
<Form> </PopupForm>,
<FormRow label={formatMessage(labels.url)}></FormRow> document.body,
<FormButtons align="center" flex>
<Button variant="primary" onClick={handleSave}>
{formatMessage(labels.save)}
</Button>
<Button onClick={handleClose}>{formatMessage(labels.cancel)}</Button>
</FormButtons>
</Form>
); );
} }

View File

@ -0,0 +1,38 @@
.menu {
width: 360px;
max-height: 300px;
overflow: auto;
}
.item {
display: flex;
flex-direction: row;
justify-content: space-between;
border-radius: var(--border-radius);
}
.item:hover {
background: var(--base75);
}
.type {
color: var(--font-color300);
}
.selected {
font-weight: bold;
}
.popup {
display: flex;
}
.filter {
display: flex;
flex-direction: column;
gap: 20px;
}
.dropdown {
min-width: 60px;
}

View File

@ -9,15 +9,13 @@ import {
PopupTrigger, PopupTrigger,
Popup, Popup,
SubmitButton, SubmitButton,
Text,
TextField, TextField,
TooltipPopup,
} from 'react-basics'; } from 'react-basics';
import Icons from 'components/icons'; import Icons from 'components/icons';
import UrlAddForm from './UrlAddForm'; import UrlAddForm from './UrlAddForm';
import { ReportContext } from 'components/pages/reports/Report'; import { ReportContext } from 'components/pages/reports/Report';
import styles from './FunnelParameters.module.css';
import BaseParameters from '../BaseParameters'; import BaseParameters from '../BaseParameters';
import ParameterList from '../ParameterList';
export function FunnelParameters() { export function FunnelParameters() {
const { report, runReport, updateReport, isRunning } = useContext(ReportContext); const { report, runReport, updateReport, isRunning } = useContext(ReportContext);
@ -28,7 +26,9 @@ export function FunnelParameters() {
const { websiteId, dateRange, urls } = parameters || {}; const { websiteId, dateRange, urls } = parameters || {};
const queryDisabled = !websiteId || !dateRange || urls?.length < 2; const queryDisabled = !websiteId || !dateRange || urls?.length < 2;
const handleSubmit = data => { const handleSubmit = (data, e) => {
e.stopPropagation();
e.preventDefault();
if (!queryDisabled) { if (!queryDisabled) {
runReport(data); runReport(data);
} }
@ -45,8 +45,23 @@ export function FunnelParameters() {
updateReport({ parameters: { urls } }); updateReport({ parameters: { urls } });
}; };
const AddUrlButton = () => {
return (
<PopupTrigger>
<Icon>
<Icons.Plus />
</Icon>
<Popup position="bottom" alignment="start">
{(close, element) => {
return <UrlAddForm element={element} onAdd={handleAddUrl} onClose={close} />;
}}
</Popup>
</PopupTrigger>
);
};
return ( return (
<Form ref={ref} values={parameters} onSubmit={handleSubmit}> <Form ref={ref} values={parameters} onSubmit={handleSubmit} preventSubmit={true}>
<BaseParameters /> <BaseParameters />
<FormRow label={formatMessage(labels.window)}> <FormRow label={formatMessage(labels.window)}>
<FormInput <FormInput
@ -56,25 +71,8 @@ export function FunnelParameters() {
<TextField autoComplete="off" /> <TextField autoComplete="off" />
</FormInput> </FormInput>
</FormRow> </FormRow>
<FormRow label={formatMessage(labels.urls)} action={<AddUrlButton onAdd={handleAddUrl} />}> <FormRow label={formatMessage(labels.urls)} action={<AddUrlButton />}>
<div className={styles.urls}> <ParameterList items={urls} onRemove={handleRemoveUrl} />
{parameters?.urls?.map((url, index) => {
return (
<div key={index} className={styles.url}>
<Text>{url}</Text>
<TooltipPopup
className={styles.icon}
label={formatMessage(labels.remove)}
position="right"
>
<Icon onClick={handleRemoveUrl.bind(null, index)}>
<Icons.Close />
</Icon>
</TooltipPopup>
</div>
);
})}
</div>
</FormRow> </FormRow>
<FormButtons> <FormButtons>
<SubmitButton variant="primary" disabled={queryDisabled} loading={isRunning}> <SubmitButton variant="primary" disabled={queryDisabled} loading={isRunning}>
@ -85,25 +83,4 @@ export function FunnelParameters() {
); );
} }
function AddUrlButton({ onAdd }) {
const { formatMessage, labels } = useMessages();
return (
<PopupTrigger>
<TooltipPopup label={formatMessage(labels.addUrl)}>
<Icon>
<Icons.Plus />
</Icon>
</TooltipPopup>
<Popup position="bottom" alignment="start">
{(close, element) => {
const { right, bottom } = element.getBoundingClientRect();
return <UrlAddForm onSave={onAdd} onClose={close} style={{ left: right, top: bottom }} />;
}}
</Popup>
</PopupTrigger>
);
}
export default FunnelParameters; export default FunnelParameters;

View File

@ -1,16 +1,15 @@
import { useState } from 'react'; import { useState } from 'react';
import { createPortal } from 'react-dom';
import { useMessages } from 'hooks'; import { useMessages } from 'hooks';
import { Button, Form, FormRow, TextField, Flexbox } from 'react-basics'; import { Button, Form, FormRow, TextField, Flexbox } from 'react-basics';
import styles from './UrlAddForm.module.css'; import styles from './UrlAddForm.module.css';
import PopupForm from '../PopupForm';
export function UrlAddForm({ defaultValue = '', style, onSave, onClose }) { export function UrlAddForm({ defaultValue = '', element, onAdd, onClose }) {
const [url, setUrl] = useState(defaultValue); const [url, setUrl] = useState(defaultValue);
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
const handleSave = e => { const handleSave = () => {
e?.stopPropagation?.(); onAdd(url);
onSave?.(url);
setUrl(''); setUrl('');
onClose(); onClose();
}; };
@ -19,29 +18,33 @@ export function UrlAddForm({ defaultValue = '', style, onSave, onClose }) {
setUrl(e.target.value); setUrl(e.target.value);
}; };
const handleClick = e => { const handleKeyDown = e => {
e.stopPropagation(); if (e.key === 'Enter') {
e.stopPropagation();
handleSave();
}
}; };
return createPortal( return (
<Form className={styles.form} onSubmit={handleSave} style={style} onClick={handleClick}> <PopupForm element={element}>
<FormRow label={formatMessage(labels.url)}> <Form>
<Flexbox gap={10}> <FormRow label={formatMessage(labels.url)}>
<TextField <Flexbox gap={10}>
className={styles.input} <TextField
name="url" className={styles.input}
value={url} value={url}
onChange={handleChange} onChange={handleChange}
autoFocus={true} autoFocus={true}
autoComplete="off" autoComplete="off"
/> onKeyDown={handleKeyDown}
<Button variant="primary" onClick={handleSave}> />
{formatMessage(labels.add)} <Button variant="primary" onClick={handleSave}>
</Button> {formatMessage(labels.add)}
</Flexbox> </Button>
</FormRow> </Flexbox>
</Form>, </FormRow>
document.body, </Form>
</PopupForm>
); );
} }

View File

@ -4,6 +4,7 @@ export * from './useCountryNames';
export * from './useDateRange'; export * from './useDateRange';
export * from './useDocumentClick'; export * from './useDocumentClick';
export * from './useEscapeKey'; export * from './useEscapeKey';
export * from './useFilters';
export * from './useForceUpdate'; export * from './useForceUpdate';
export * from './useLanguageNames'; export * from './useLanguageNames';
export * from './useLocale'; export * from './useLocale';

32
hooks/useFilters.js Normal file
View File

@ -0,0 +1,32 @@
import { useMessages } from 'hooks';
export function useFilters() {
const { formatMessage, labels } = useMessages();
const filters = {
eq: formatMessage(labels.equals),
neq: formatMessage(labels.doesNotEqual),
c: formatMessage(labels.contains),
dnc: formatMessage(labels.doesNotContain),
t: formatMessage(labels.true),
f: formatMessage(labels.false),
gt: formatMessage(labels.greaterThan),
lt: formatMessage(labels.lessThan),
gte: formatMessage(labels.greaterThanEquals),
lte: formatMessage(labels.lessThanEquals),
be: formatMessage(labels.before),
af: formatMessage(labels.after),
};
const types = {
string: ['eq', 'neq'],
array: ['c', 'dnc'],
boolean: ['t', 'f'],
number: ['eq', 'neq', 'gt', 'lt', 'gte', 'lte'],
date: ['be', 'af'],
};
return { filters, types };
}
export default useFilters;

View File

@ -16,6 +16,14 @@ export function useReport(reportId, defaultParameters) {
const loadReport = async id => { const loadReport = async id => {
const data = await get(`/reports/${id}`); const data = await get(`/reports/${id}`);
const { dateRange } = data?.parameters || {};
const { startDate, endDate } = dateRange || {};
if (startDate && endDate) {
dateRange.startDate = new Date(startDate);
dateRange.endDate = new Date(endDate);
}
setReport(data); setReport(data);
}; };

View File

@ -62,7 +62,7 @@ export const EVENT_TYPE = {
customEvent: 2, customEvent: 2,
} as const; } as const;
export const DYNAMIC_DATA_TYPE = { export const DATA_TYPE = {
string: 1, string: 1,
number: 2, number: 2,
boolean: 3, boolean: 3,
@ -71,11 +71,11 @@ export const DYNAMIC_DATA_TYPE = {
} as const; } as const;
export const DATA_TYPES = { export const DATA_TYPES = {
[DYNAMIC_DATA_TYPE.string]: 'string', [DATA_TYPE.string]: 'string',
[DYNAMIC_DATA_TYPE.number]: 'number', [DATA_TYPE.number]: 'number',
[DYNAMIC_DATA_TYPE.boolean]: 'boolean', [DATA_TYPE.boolean]: 'boolean',
[DYNAMIC_DATA_TYPE.date]: 'date', [DATA_TYPE.date]: 'date',
[DYNAMIC_DATA_TYPE.array]: 'array', [DATA_TYPE.array]: 'array',
}; };
export const KAFKA_TOPIC = { export const KAFKA_TOPIC = {

View File

@ -1,5 +1,5 @@
import { isValid, parseISO } from 'date-fns'; import { isValid, parseISO } from 'date-fns';
import { DYNAMIC_DATA_TYPE } from './constants'; import { DATA_TYPE } from './constants';
import { DynamicDataType } from './types'; import { DynamicDataType } from './types';
export function flattenJSON( export function flattenJSON(
@ -42,24 +42,24 @@ function createKey(key, value, acc: { keyValues: any[]; parentKey: string }) {
switch (type) { switch (type) {
case 'number': case 'number':
dynamicDataType = DYNAMIC_DATA_TYPE.number; dynamicDataType = DATA_TYPE.number;
break; break;
case 'string': case 'string':
dynamicDataType = DYNAMIC_DATA_TYPE.string; dynamicDataType = DATA_TYPE.string;
break; break;
case 'boolean': case 'boolean':
dynamicDataType = DYNAMIC_DATA_TYPE.boolean; dynamicDataType = DATA_TYPE.boolean;
value = value ? 'true' : 'false'; value = value ? 'true' : 'false';
break; break;
case 'date': case 'date':
dynamicDataType = DYNAMIC_DATA_TYPE.date; dynamicDataType = DATA_TYPE.date;
break; break;
case 'object': case 'object':
dynamicDataType = DYNAMIC_DATA_TYPE.array; dynamicDataType = DATA_TYPE.array;
value = JSON.stringify(value); value = JSON.stringify(value);
break; break;
default: default:
dynamicDataType = DYNAMIC_DATA_TYPE.string; dynamicDataType = DATA_TYPE.string;
break; break;
} }

View File

@ -1,5 +1,5 @@
import { NextApiRequest } from 'next'; import { NextApiRequest } from 'next';
import { COLLECTION_TYPE, DYNAMIC_DATA_TYPE, EVENT_TYPE, KAFKA_TOPIC, ROLES } from './constants'; import { COLLECTION_TYPE, DATA_TYPE, EVENT_TYPE, KAFKA_TOPIC, ROLES } from './constants';
type ObjectValues<T> = T[keyof T]; type ObjectValues<T> = T[keyof T];
@ -9,7 +9,7 @@ export type Role = ObjectValues<typeof ROLES>;
export type EventType = ObjectValues<typeof EVENT_TYPE>; export type EventType = ObjectValues<typeof EVENT_TYPE>;
export type DynamicDataType = ObjectValues<typeof DYNAMIC_DATA_TYPE>; export type DynamicDataType = ObjectValues<typeof DATA_TYPE>;
export type KafkaTopic = ObjectValues<typeof KAFKA_TOPIC>; export type KafkaTopic = ObjectValues<typeof KAFKA_TOPIC>;

View File

@ -1,20 +1,24 @@
import { useRouter } from 'next/router'; import { useRouter } from 'next/router';
import AppLayout from 'components/layout/AppLayout'; import AppLayout from 'components/layout/AppLayout';
import FunnelReport from 'components/pages/reports/funnel/FunnelReport'; import ReportDetails from 'components/pages/reports/ReportDetails';
import useMessages from 'hooks/useMessages'; import { useApi, useMessages } from 'hooks';
export default function ReportsPage() { export default function ReportsPage() {
const { formatMessage, labels } = useMessages(); const { formatMessage, labels } = useMessages();
const router = useRouter(); const router = useRouter();
const { id } = router.query; const { id } = router.query;
const { get, useQuery } = useApi();
const { data: report } = useQuery(['reports', id], () => get(`/reports/${id}`), {
enabled: !!id,
});
if (!id) { if (!id || !report) {
return null; return null;
} }
return ( return (
<AppLayout title={formatMessage(labels.websites)}> <AppLayout title={formatMessage(labels.websites)}>
<FunnelReport reportId={id} /> <ReportDetails reportId={report.id} reportType={report.type} />
</AppLayout> </AppLayout>
); );
} }

View File

@ -230,7 +230,7 @@
"label.edit": [ "label.edit": [
{ {
"type": 0, "type": 0,
"value": "Редактировать" "value": "Изменить"
} }
], ],
"label.edit-dashboard": [ "label.edit-dashboard": [
@ -820,15 +820,7 @@
"message.delete-website": [ "message.delete-website": [
{ {
"type": 0, "type": 0,
"value": "To delete this website, type " "value": "Для удаления введите DELETE"
},
{
"type": 1,
"value": "confirmation"
},
{
"type": 0,
"value": " in the box below to confirm."
} }
], ],
"message.delete-website-warning": [ "message.delete-website-warning": [
@ -922,15 +914,7 @@
"message.reset-website": [ "message.reset-website": [
{ {
"type": 0, "type": 0,
"value": "To reset this website, type " "value": "Для сброса введите RESET"
},
{
"type": 1,
"value": "confirmation"
},
{
"type": 0,
"value": " in the box below to confirm."
} }
], ],
"message.reset-website-warning": [ "message.reset-website-warning": [

View File

@ -14,7 +14,7 @@
"label.activity-log": [ "label.activity-log": [
{ {
"type": 0, "type": 0,
"value": "Activity log" "value": "Aktivitetslogg"
} }
], ],
"label.add-website": [ "label.add-website": [
@ -44,7 +44,7 @@
"label.analytics": [ "label.analytics": [
{ {
"type": 0, "type": 0,
"value": "Analytics" "value": "Analys"
} }
], ],
"label.average-visit-time": [ "label.average-visit-time": [
@ -86,19 +86,19 @@
"label.cities": [ "label.cities": [
{ {
"type": 0, "type": 0,
"value": "Cities" "value": "Städer"
} }
], ],
"label.clear-all": [ "label.clear-all": [
{ {
"type": 0, "type": 0,
"value": "Clear all" "value": "Rensa alla"
} }
], ],
"label.confirm": [ "label.confirm": [
{ {
"type": 0, "type": 0,
"value": "Confirm" "value": "Bekräfta"
} }
], ],
"label.confirm-password": [ "label.confirm-password": [
@ -110,7 +110,7 @@
"label.continue": [ "label.continue": [
{ {
"type": 0, "type": 0,
"value": "Continue" "value": "Fortsätt"
} }
], ],
"label.countries": [ "label.countries": [
@ -122,19 +122,19 @@
"label.create-team": [ "label.create-team": [
{ {
"type": 0, "type": 0,
"value": "Create team" "value": "Skapa team"
} }
], ],
"label.create-user": [ "label.create-user": [
{ {
"type": 0, "type": 0,
"value": "Create user" "value": "Skapa användare"
} }
], ],
"label.created": [ "label.created": [
{ {
"type": 0, "type": 0,
"value": "Created" "value": "Skapad"
} }
], ],
"label.current-password": [ "label.current-password": [
@ -182,13 +182,13 @@
"label.delete-team": [ "label.delete-team": [
{ {
"type": 0, "type": 0,
"value": "Delete team" "value": "Radera team"
} }
], ],
"label.delete-user": [ "label.delete-user": [
{ {
"type": 0, "type": 0,
"value": "Delete user" "value": "Radera användare"
} }
], ],
"label.delete-website": [ "label.delete-website": [
@ -206,7 +206,7 @@
"label.details": [ "label.details": [
{ {
"type": 0, "type": 0,
"value": "Details" "value": "Detailjer"
} }
], ],
"label.devices": [ "label.devices": [
@ -236,7 +236,7 @@
"label.edit-dashboard": [ "label.edit-dashboard": [
{ {
"type": 0, "type": 0,
"value": "Edit dashboard" "value": "Redigera översikt"
} }
], ],
"label.enable-share-url": [ "label.enable-share-url": [
@ -278,13 +278,13 @@
"label.join": [ "label.join": [
{ {
"type": 0, "type": 0,
"value": "Join" "value": "Gå med"
} }
], ],
"label.join-team": [ "label.join-team": [
{ {
"type": 0, "type": 0,
"value": "Join team" "value": "gå med i team"
} }
], ],
"label.language": [ "label.language": [
@ -336,13 +336,13 @@
"label.leave": [ "label.leave": [
{ {
"type": 0, "type": 0,
"value": "Leave" "value": "Lämna"
} }
], ],
"label.leave-team": [ "label.leave-team": [
{ {
"type": 0, "type": 0,
"value": "Leave team" "value": "Lämna team"
} }
], ],
"label.login": [ "label.login": [
@ -360,7 +360,7 @@
"label.members": [ "label.members": [
{ {
"type": 0, "type": 0,
"value": "Members" "value": "Medlemmar"
} }
], ],
"label.mobile": [ "label.mobile": [
@ -390,7 +390,7 @@
"label.none": [ "label.none": [
{ {
"type": 0, "type": 0,
"value": "None" "value": "Inga"
} }
], ],
"label.operating-systems": [ "label.operating-systems": [
@ -442,19 +442,19 @@
"label.queries": [ "label.queries": [
{ {
"type": 0, "type": 0,
"value": "Queries" "value": "Frågor"
} }
], ],
"label.query": [ "label.query": [
{ {
"type": 0, "type": 0,
"value": "Query" "value": "Frågor"
} }
], ],
"label.query-parameters": [ "label.query-parameters": [
{ {
"type": 0, "type": 0,
"value": "Query parameters" "value": "Fråge-parametrar"
} }
], ],
"label.realtime": [ "label.realtime": [
@ -478,19 +478,19 @@
"label.regenerate": [ "label.regenerate": [
{ {
"type": 0, "type": 0,
"value": "Regenerate" "value": "Regenerera"
} }
], ],
"label.regions": [ "label.regions": [
{ {
"type": 0, "type": 0,
"value": "Regions" "value": "Regioner"
} }
], ],
"label.remove": [ "label.remove": [
{ {
"type": 0, "type": 0,
"value": "Remove" "value": "Ta bort"
} }
], ],
"label.reports": [ "label.reports": [
@ -520,7 +520,7 @@
"label.role": [ "label.role": [
{ {
"type": 0, "type": 0,
"value": "Role" "value": "Roll"
} }
], ],
"label.save": [ "label.save": [
@ -532,7 +532,7 @@
"label.screens": [ "label.screens": [
{ {
"type": 0, "type": 0,
"value": "Screens" "value": "Upplösning"
} }
], ],
"label.select-date": [ "label.select-date": [
@ -544,7 +544,7 @@
"label.select-website": [ "label.select-website": [
{ {
"type": 0, "type": 0,
"value": "Select website" "value": "Välj webbsajt"
} }
], ],
"label.sessions": [ "label.sessions": [
@ -586,7 +586,7 @@
"label.team-guest": [ "label.team-guest": [
{ {
"type": 0, "type": 0,
"value": "Team guest" "value": "Team-gäst"
} }
], ],
"label.team-id": [ "label.team-id": [
@ -598,19 +598,19 @@
"label.team-member": [ "label.team-member": [
{ {
"type": 0, "type": 0,
"value": "Team member" "value": "Team-medlem"
} }
], ],
"label.team-owner": [ "label.team-owner": [
{ {
"type": 0, "type": 0,
"value": "Team owner" "value": "Team-ägare"
} }
], ],
"label.teams": [ "label.teams": [
{ {
"type": 0, "type": 0,
"value": "Teams" "value": "Team"
} }
], ],
"label.theme": [ "label.theme": [
@ -646,7 +646,7 @@
"label.title": [ "label.title": [
{ {
"type": 0, "type": 0,
"value": "Title" "value": "Titel"
} }
], ],
"label.today": [ "label.today": [
@ -688,7 +688,7 @@
"label.user": [ "label.user": [
{ {
"type": 0, "type": 0,
"value": "User" "value": "Användare"
} }
], ],
"label.username": [ "label.username": [
@ -706,7 +706,7 @@
"label.view": [ "label.view": [
{ {
"type": 0, "type": 0,
"value": "View" "value": "Visa"
} }
], ],
"label.view-details": [ "label.view-details": [
@ -736,7 +736,7 @@
"label.website-id": [ "label.website-id": [
{ {
"type": 0, "type": 0,
"value": "Website ID" "value": "Webbsajt-ID"
} }
], ],
"label.websites": [ "label.websites": [
@ -748,7 +748,7 @@
"label.yesterday": [ "label.yesterday": [
{ {
"type": 0, "type": 0,
"value": "Yesterday" "value": "Igår"
} }
], ],
"message.active-users": [ "message.active-users": [
@ -806,7 +806,7 @@
"message.confirm-leave": [ "message.confirm-leave": [
{ {
"type": 0, "type": 0,
"value": "Are you sure you want to leave " "value": "Är du säker på att du vill lämna "
}, },
{ {
"type": 1, "type": 1,
@ -878,7 +878,7 @@
}, },
{ {
"type": 0, "type": 0,
"value": " on " "value": " "
}, },
{ {
"type": 1, "type": 1,
@ -906,7 +906,7 @@
"message.min-password-length": [ "message.min-password-length": [
{ {
"type": 0, "type": 0,
"value": "Minimum length of " "value": "Minst "
}, },
{ {
"type": 1, "type": 1,
@ -914,7 +914,7 @@
}, },
{ {
"type": 0, "type": 0,
"value": " characters" "value": " tecken"
} }
], ],
"message.no-data-available": [ "message.no-data-available": [
@ -932,13 +932,13 @@
"message.no-teams": [ "message.no-teams": [
{ {
"type": 0, "type": 0,
"value": "You have not created any teams." "value": "Du har inte skapat några team."
} }
], ],
"message.no-users": [ "message.no-users": [
{ {
"type": 0, "type": 0,
"value": "There are no users." "value": "Det finns inga användare."
} }
], ],
"message.page-not-found": [ "message.page-not-found": [
@ -950,7 +950,7 @@
"message.reset-website": [ "message.reset-website": [
{ {
"type": 0, "type": 0,
"value": "To reset this website, type " "value": "För att återställa statistiken skriv "
}, },
{ {
"type": 1, "type": 1,
@ -958,7 +958,7 @@
}, },
{ {
"type": 0, "type": 0,
"value": " in the box below to confirm." "value": " i rutan nedan."
} }
], ],
"message.reset-website-warning": [ "message.reset-website-warning": [
@ -990,13 +990,13 @@
"message.team-already-member": [ "message.team-already-member": [
{ {
"type": 0, "type": 0,
"value": "You are already a member of the team." "value": "Du är redan medlem i teamet."
} }
], ],
"message.team-not-found": [ "message.team-not-found": [
{ {
"type": 0, "type": 0,
"value": "Team not found." "value": "Team kan inte hittas."
} }
], ],
"message.tracking-code": [ "message.tracking-code": [
@ -1008,7 +1008,7 @@
"message.user-deleted": [ "message.user-deleted": [
{ {
"type": 0, "type": 0,
"value": "User deleted." "value": "Användare raderad."
} }
], ],
"message.visitor-log": [ "message.visitor-log": [
@ -1054,7 +1054,7 @@
"messages.no-team-websites": [ "messages.no-team-websites": [
{ {
"type": 0, "type": 0,
"value": "This team does not have any websites." "value": "Det här teamet har inga webbsajter."
} }
], ],
"messages.no-websites-configured": [ "messages.no-websites-configured": [
@ -1066,7 +1066,7 @@
"messages.team-websites-info": [ "messages.team-websites-info": [
{ {
"type": 0, "type": 0,
"value": "Websites can be viewed by anyone on the team." "value": "Websajter kan ses av alla i teamet."
} }
] ]
} }

View File

@ -26,7 +26,8 @@ async function relationalQuery(websiteId: string, startDate: Date, endDate: Date
from event_data from event_data
where website_id = $1${toUuid()} where website_id = $1${toUuid()}
and created_at >= $2 and created_at >= $2
and created_at between $3 and $4`, and created_at between $3 and $4
order by event_key asc`,
params, params,
); );
} }
@ -43,7 +44,8 @@ async function clickhouseQuery(websiteId: string, startDate: Date, endDate: Date
from event_data from event_data
where website_id = {websiteId:UUID} where website_id = {websiteId:UUID}
and created_at >= ${getDateFormat(resetDate)} and created_at >= ${getDateFormat(resetDate)}
and ${getBetweenDates('created_at', startDate, endDate)}`, and ${getBetweenDates('created_at', startDate, endDate)}
order by event_key asc`,
params, params,
); );
} }

View File

@ -1,5 +1,5 @@
import { Prisma } from '@prisma/client'; import { Prisma } from '@prisma/client';
import { DYNAMIC_DATA_TYPE } from 'lib/constants'; import { DATA_TYPE } from 'lib/constants';
import { uuid } from 'lib/crypto'; import { uuid } from 'lib/crypto';
import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db'; import { CLICKHOUSE, PRISMA, runQuery } from 'lib/db';
import { flattenJSON } from 'lib/dynamicData'; import { flattenJSON } from 'lib/dynamicData';
@ -38,13 +38,13 @@ async function relationalQuery(data: {
websiteId, websiteId,
key: a.key, key: a.key,
stringValue: stringValue:
a.dynamicDataType === DYNAMIC_DATA_TYPE.string || a.dynamicDataType === DATA_TYPE.string ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.boolean || a.dynamicDataType === DATA_TYPE.boolean ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.array a.dynamicDataType === DATA_TYPE.array
? a.value ? a.value
: null, : null,
numericValue: a.dynamicDataType === DYNAMIC_DATA_TYPE.number ? a.value : null, numericValue: a.dynamicDataType === DATA_TYPE.number ? a.value : null,
dateValue: a.dynamicDataType === DYNAMIC_DATA_TYPE.date ? new Date(a.value) : null, dateValue: a.dynamicDataType === DATA_TYPE.date ? new Date(a.value) : null,
dataType: a.dynamicDataType, dataType: a.dynamicDataType,
})); }));
@ -76,13 +76,13 @@ async function clickhouseQuery(data: {
event_name: eventName, event_name: eventName,
event_key: a.key, event_key: a.key,
string_value: string_value:
a.dynamicDataType === DYNAMIC_DATA_TYPE.string || a.dynamicDataType === DATA_TYPE.string ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.boolean || a.dynamicDataType === DATA_TYPE.boolean ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.array a.dynamicDataType === DATA_TYPE.array
? a.value ? a.value
: null, : null,
numeric_value: a.dynamicDataType === DYNAMIC_DATA_TYPE.number ? a.value : null, numeric_value: a.dynamicDataType === DATA_TYPE.number ? a.value : null,
date_value: a.dynamicDataType === DYNAMIC_DATA_TYPE.date ? getDateFormat(a.value) : null, date_value: a.dynamicDataType === DATA_TYPE.date ? getDateFormat(a.value) : null,
data_type: a.dynamicDataType, data_type: a.dynamicDataType,
created_at: createdAt, created_at: createdAt,
})); }));

View File

@ -1,4 +1,4 @@
import { DYNAMIC_DATA_TYPE } from 'lib/constants'; import { DATA_TYPE } from 'lib/constants';
import { uuid } from 'lib/crypto'; import { uuid } from 'lib/crypto';
import { flattenJSON } from 'lib/dynamicData'; import { flattenJSON } from 'lib/dynamicData';
import prisma from 'lib/prisma'; import prisma from 'lib/prisma';
@ -20,13 +20,13 @@ export async function saveSessionData(data: {
sessionId, sessionId,
key: a.key, key: a.key,
stringValue: stringValue:
a.dynamicDataType === DYNAMIC_DATA_TYPE.string || a.dynamicDataType === DATA_TYPE.string ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.boolean || a.dynamicDataType === DATA_TYPE.boolean ||
a.dynamicDataType === DYNAMIC_DATA_TYPE.array a.dynamicDataType === DATA_TYPE.array
? a.value ? a.value
: null, : null,
numericValue: a.dynamicDataType === DYNAMIC_DATA_TYPE.number ? a.value : null, numericValue: a.dynamicDataType === DATA_TYPE.number ? a.value : null,
dateValue: a.dynamicDataType === DYNAMIC_DATA_TYPE.date ? new Date(a.value) : null, dateValue: a.dynamicDataType === DATA_TYPE.date ? new Date(a.value) : null,
dataType: a.dynamicDataType, dataType: a.dynamicDataType,
})); }));