Files
stats/apps/start/src/components/sessions/table/columns.tsx
Carl-Gerhard Lindesvärd 11e9ecac1a feat: group analytics
* wip

* wip

* wip

* wip

* wip

* add buffer

* wip

* wip

* fixes

* fix

* wip

* group validation

* fix group issues

* docs: add groups
2026-03-20 10:46:09 +01:00

298 lines
8.0 KiB
TypeScript

import { round } from '@openpanel/common';
import type { IServiceSession } from '@openpanel/db';
import type { ColumnDef } from '@tanstack/react-table';
import { Video } from 'lucide-react';
import { ColumnCreatedAt } from '@/components/column-created-at';
import { ProjectLink } from '@/components/links';
import { ProfileAvatar } from '@/components/profiles/profile-avatar';
import { SerieIcon } from '@/components/report-chart/common/serie-icon';
import { getProfileName } from '@/utils/getters';
function formatDuration(milliseconds: number): string {
const seconds = milliseconds / 1000;
if (seconds < 60) {
return `${round(seconds, 1)}s`;
}
const minutes = Math.floor(seconds / 60);
const remainingSeconds = seconds % 60;
if (minutes < 60) {
return remainingSeconds > 0
? `${minutes}m ${round(remainingSeconds, 0)}s`
: `${minutes}m`;
}
const hours = Math.floor(minutes / 60);
const remainingMinutes = minutes % 60;
return `${hours}h ${round(remainingMinutes, 0)}m`;
}
export function useColumns() {
const columns: ColumnDef<IServiceSession>[] = [
{
accessorKey: 'createdAt',
header: 'Started',
size: ColumnCreatedAt.size,
cell: ({ row }) => {
const item = row.original;
return <ColumnCreatedAt>{item.createdAt}</ColumnCreatedAt>;
},
},
{
accessorKey: 'id',
header: 'Session ID',
size: 120,
cell: ({ row }) => {
const session = row.original;
return (
<div className="row items-center gap-2">
<ProjectLink
className="font-medium"
href={`/sessions/${session.id}`}
title={session.id}
>
{session.id.slice(0, 8)}...
</ProjectLink>
{session.hasReplay && (
<ProjectLink
aria-label="View replay"
className="text-muted-foreground hover:text-foreground"
href={`/sessions/${session.id}#replay`}
title="View replay"
>
<Video className="size-4" />
</ProjectLink>
)}
</div>
);
},
},
{
accessorKey: 'profileId',
header: 'Profile',
size: 150,
cell: ({ row }) => {
const session = row.original;
if (session.profile) {
return (
<ProjectLink
className="row items-center gap-2 font-medium hover:underline"
href={`/profiles/${encodeURIComponent(session.profile.id)}`}
>
<ProfileAvatar size="sm" {...session.profile} />
{getProfileName(session.profile)}
</ProjectLink>
);
}
return (
<ProjectLink
className="font-medium font-mono"
href={`/profiles/${encodeURIComponent(session.profileId)}`}
>
{session.profileId}
</ProjectLink>
);
},
},
{
accessorKey: 'entryPath',
header: 'Entry Page',
size: 200,
cell: ({ row }) => {
const session = row.original;
return (
<div className="min-w-0">
<span className="font-mono">{session.entryPath || '/'}</span>
</div>
);
},
},
{
accessorKey: 'exitPath',
header: 'Exit Page',
size: 200,
cell: ({ row }) => {
const session = row.original;
return (
<div className="min-w-0">
<span className="font-mono">
{session.exitPath || session.entryPath || '/'}
</span>
</div>
);
},
},
{
accessorKey: 'duration',
header: 'Duration',
size: 100,
cell: ({ row }) => {
const session = row.original;
return (
<div className="font-medium">{formatDuration(session.duration)}</div>
);
},
},
{
accessorKey: 'isBounce',
header: 'Bounce',
size: 80,
cell: ({ row }) => {
const session = row.original;
return (
<div className="text-center">
{session.isBounce ? (
<span className="text-orange-600">Yes</span>
) : (
<span className="text-green-600">No</span>
)}
</div>
);
},
},
{
accessorKey: 'referrerName',
header: 'Referrer',
size: 150,
cell: ({ row }) => {
const session = row.original;
const ref = session.referrerName || session.referrer || 'Direct';
return (
<div className="flex min-w-0 items-center gap-2">
<SerieIcon name={ref} />
<span className="truncate">{ref}</span>
</div>
);
},
},
{
accessorKey: 'country',
header: 'Location',
size: 150,
cell: ({ row }) => {
const session = row.original;
return (
<div className="flex min-w-0 items-center gap-2">
<SerieIcon name={session.country} />
<span className="truncate">{session.city || session.country}</span>
</div>
);
},
},
{
accessorKey: 'os',
header: 'OS',
size: 120,
cell: ({ row }) => {
const session = row.original;
return (
<div className="flex min-w-0 items-center gap-2">
<SerieIcon name={session.os} />
<span className="truncate">{session.os}</span>
</div>
);
},
},
{
accessorKey: 'browser',
header: 'Browser',
size: 120,
cell: ({ row }) => {
const session = row.original;
return (
<div className="flex min-w-0 items-center gap-2">
<SerieIcon name={session.browser} />
<span className="truncate">{session.browser}</span>
</div>
);
},
},
{
accessorKey: 'device',
header: 'Device',
size: 150,
cell: ({ row }) => {
const session = row.original;
let deviceInfo =
session.brand || session.model
? [session.brand, session.model].filter(Boolean).join(' / ')
: session.device;
if (deviceInfo === 'K') {
deviceInfo = session.device;
}
return (
<div className="flex min-w-0 items-center gap-2">
<SerieIcon name={session.device} />
<span className="truncate">{deviceInfo}</span>
</div>
);
},
},
{
accessorKey: 'screenViewCount',
header: 'Page views',
size: 100,
cell: ({ row }) => {
const session = row.original;
return (
<div className="text-center font-medium">
{session.screenViewCount}
</div>
);
},
},
{
accessorKey: 'eventCount',
header: 'Events',
size: 90,
cell: ({ row }) => {
const session = row.original;
return (
<div className="text-center font-medium">{session.eventCount}</div>
);
},
},
{
accessorKey: 'revenue',
header: 'Revenue',
size: 100,
cell: ({ row }) => {
const session = row.original;
return session.revenue > 0 ? (
<div className="font-medium text-green-600">
${session.revenue.toFixed(2)}
</div>
) : (
<div className="text-muted-foreground">-</div>
);
},
},
{
accessorKey: 'deviceId',
header: 'Device ID',
size: 120,
},
{
accessorKey: 'groups',
header: 'Groups',
size: 200,
cell: ({ row }) => {
const { groups } = row.original;
if (!groups?.length) return null;
return (
<div className="flex flex-wrap gap-1">
{groups.map((g) => (
<span
key={g}
className="rounded bg-muted px-1.5 py-0.5 text-xs font-mono"
>
{g}
</span>
))}
</div>
);
},
},
];
return columns;
}