Skip to content

Commit 8e6d985

Browse files
atrakhConvex, Inc.
authored and
Convex, Inc.
committed
dashboard: improved empty states for data, functions, health pages (#36222)
- New overlay for the health page when there has never been a deployment - New empty state for the data page showing an empty table with blank data - New empty state for the functions page showing an example of the charts - New UI for "Select a file to open in it" text on the functions page - ResizeHandle now has a title that appears when it is closed - SidebarDetailLayout panels are now collapsed by default if the available width is too small (<768px) <img width="1145" alt="Screenshot 2025-04-04 at 3 09 29 PM" src="https://github.com/user-attachments/assets/24f4a93c-849b-46a9-995b-7122f498a17f" /> <img width="1145" alt="Screenshot 2025-04-04 at 3 09 21 PM" src="https://github.com/user-attachments/assets/4e33764d-d033-4867-b298-96c8f2bb030f" /> <img width="1145" alt="Screenshot 2025-04-04 at 3 09 18 PM" src="https://github.com/user-attachments/assets/51fac801-30dc-434e-ae4c-1bd7664d3b2e" /> <img width="1145" alt="Screenshot 2025-04-04 at 3 09 13 PM" src="https://github.com/user-attachments/assets/1365b1da-be43-4f43-9a71-5607f3852ccb" /> <img width="1145" alt="Screenshot 2025-04-04 at 3 09 10 PM" src="https://github.com/user-attachments/assets/a5ad7fa2-20b4-414c-b8f8-30be30e6c579" /> GitOrigin-RevId: 2f78be282b410b6057dc2db3358a418467aa45b2
1 parent 5a8e3a9 commit 8e6d985

File tree

18 files changed

+519
-109
lines changed

18 files changed

+519
-109
lines changed

npm-packages/dashboard-common/src/elements/EmptySection.tsx

+3-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ export function EmptySection({
1414
sheet = true,
1515
}: {
1616
Icon: React.FunctionComponent<{ className: string | undefined }>;
17-
color?: "yellow" | "red" | "purple" | "green";
17+
color?: "yellow" | "red" | "purple" | "green" | "none";
1818
header: string;
1919
body: React.ReactNode;
2020
action?: React.ReactNode;
@@ -35,11 +35,12 @@ export function EmptySection({
3535
"from-red-300 to-util-brand-red",
3636
color === "purple" && "from-purple-200 to-util-brand-purple",
3737
color === "green" && "from-util-success to-util-success",
38+
color === "none" && "bg-transparent shadow-none",
3839
)}
3940
>
4041
<Icon className="h-6 w-6 text-white" />
4142
</div>
42-
<h2 className="mx-2 mb-2 text-content-secondary">{header}</h2>
43+
<h2 className="mx-2 mb-2">{header}</h2>
4344

4445
<p className="mb-2 max-w-prose text-balance text-content-tertiary">
4546
{body}

npm-packages/dashboard-common/src/elements/PageContent.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import { ReactNode } from "react";
22

33
export function PageContent({ children }: { children: ReactNode }) {
44
return (
5-
<div className="h-full min-w-0 flex-1 overflow-x-auto bg-background-primary">
5+
<div className="relative h-full min-w-0 flex-1 overflow-x-auto bg-background-primary">
66
{children}
77
</div>
88
);

npm-packages/dashboard-common/src/elements/Tab.tsx

+3-5
Original file line numberDiff line numberDiff line change
@@ -20,15 +20,13 @@ export function Tab({
2020
tip={tip}
2121
variant="unstyled"
2222
className={cn(
23-
"p-2 text-sm rounded whitespace-nowrap cursor-pointer",
23+
"px-3 py-2 text-sm whitespace-nowrap cursor-pointer",
2424
!disabled && selected
25-
? "text-content-primary"
25+
? "text-content-primary border-b-2 border-content-primary"
2626
: "text-content-secondary",
2727
disabled
2828
? "disabled:text-content-secondary cursor-not-allowed"
29-
: "hover:bg-background-tertiary",
30-
selected &&
31-
"font-semibold underline underline-offset-8 decoration-2",
29+
: "hover:text-content-primary",
3230
// It's OK for tabs.
3331
// eslint-disable-next-line no-restricted-syntax
3432
large && "text-lg",

npm-packages/dashboard-common/src/features/data/components/DataContent.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -190,7 +190,7 @@ export function DataContent({
190190
<PanelGroup
191191
direction="horizontal"
192192
className={cn(
193-
"flex w-full h-full overflow-x-auto scrollbar pl-6 min-w-[30rem]",
193+
"flex w-full h-full overflow-x-auto scrollbar pl-6 min-w-[20rem]",
194194
popupEl ? "pr-0" : "pr-6",
195195
)}
196196
autoSaveId="data-content"
@@ -199,7 +199,7 @@ export function DataContent({
199199
className={cn(
200200
"flex shrink flex-col gap-2 overflow-hidden py-4",
201201
"max-w-full",
202-
popupEl ? "min-w-[10rem]" : "min-w-[30rem]",
202+
popupEl ? "min-w-[10rem]" : "min-w-[20rem]",
203203
)}
204204
ref={ref}
205205
defaultSize={80}

npm-packages/dashboard-common/src/features/data/components/DataSidebar.tsx

+3-5
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@ export function DataSidebar({
6262
/>
6363
)}
6464
<div className={cn("flex-1 overflow-auto scrollbar px-3 pt-1")}>
65-
<div className="flex flex-col gap-1">
65+
<div className="flex flex-col gap-0.5">
6666
{Array.from(tables.keys())
6767
.filter(
6868
(r) =>
@@ -93,9 +93,7 @@ export function DataSidebar({
9393
icon={<CodeIcon />}
9494
className="animate-fadeInFromLoading overflow-hidden"
9595
>
96-
<span className="truncate">
97-
{showSchema.hasSaved ? <>Show Schema</> : <>Generate Schema</>}
98-
</span>
96+
<span className="truncate">Schema</span>
9997
</Button>
10098
)}
10199
</div>
@@ -194,11 +192,11 @@ export function CreateNewTable({ tableData }: { tableData: TableMetadata }) {
194192
</form>
195193
) : (
196194
<Button
197-
inline
198195
size="sm"
199196
className="mt-1 max-w-full"
200197
onClick={() => setNewTableName("")}
201198
icon={<PlusIcon aria-hidden="true" />}
199+
inline
202200
disabled={
203201
!canCreateTable || !!(selectedNent && selectedNent.state !== "active")
204202
}

npm-packages/dashboard-common/src/features/data/components/DataView.tsx

+2
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,7 @@ export function DataView() {
8484
<div className="h-full w-full">
8585
<div className="flex h-full">
8686
<SidebarDetailLayout
87+
resizeHandleTitle="Tables"
8788
panelSizeKey={`${deploymentId}/data`}
8889
sidebarComponent={<DataSideBarSkeleton />}
8990
contentComponent={<DataContentSkeleton />}
@@ -101,6 +102,7 @@ export function DataView() {
101102
showSchema={showSchemaProps}
102103
/>
103104
}
105+
resizeHandleTitle="Tables"
104106
contentComponent={
105107
tableMetadata.name === null ? (
106108
<EmptyData />
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,10 @@
1-
import { PlusIcon, TableIcon } from "@radix-ui/react-icons";
1+
import {
2+
PlusIcon,
3+
TableIcon,
4+
MixerHorizontalIcon,
5+
ChevronDownIcon,
6+
DotsVerticalIcon,
7+
} from "@radix-ui/react-icons";
28
import { useContext } from "react";
39
import { CreateNewTable } from "@common/features/data/components/DataSidebar";
410
import { EmptySection } from "@common/elements/EmptySection";
@@ -9,9 +15,12 @@ import { Loading } from "@common/elements/Loading";
915
import { Button } from "@common/elements/Button";
1016
import { Sheet } from "@common/elements/Sheet";
1117

18+
// Example table data for the background
19+
const EXAMPLE_COLUMNS = ["_id", "name", "email", "_creationTime"];
20+
1221
export function EmptyData() {
1322
return (
14-
<div className="flex h-full p-6">
23+
<div className="flex h-full items-center justify-center p-6">
1524
<EmptyDataContent noTables />
1625
</div>
1726
);
@@ -45,54 +54,159 @@ export function EmptyDataContent({
4554
}
4655

4756
return (
48-
<Sheet padding={false} className="w-full">
49-
<EmptySection
50-
Icon={TableIcon}
51-
header={
52-
noTables ? "There are no tables here yet." : "This table is empty."
53-
}
54-
sheet={false}
55-
body={
56-
noTables
57-
? "Create a table to start storing data."
58-
: "Create a document or run a mutation to start storing data."
59-
}
60-
action={
61-
noTables ? (
62-
<CreateNewTable tableData={tableMetadata} />
63-
) : (
64-
<>
65-
{openAddDocuments && (
66-
<Button
67-
inline
68-
onClick={() => {
69-
log("open add documents panel", { how: "empty data" });
70-
openAddDocuments();
71-
}}
72-
size="sm"
73-
disabled={
74-
!canAddDocuments ||
75-
!!(selectedNent && selectedNent.state !== "active")
76-
}
77-
tip={
78-
selectedNent && selectedNent.state !== "active"
79-
? "Cannot add documents in an unmounted component."
80-
: !canAddDocuments &&
81-
"You do not have permission to add documents in production."
82-
}
83-
icon={<PlusIcon aria-hidden="true" />}
84-
>
85-
Add Documents
86-
</Button>
87-
)}
88-
</>
89-
)
90-
}
91-
learnMoreButton={{
92-
href: "https://docs.convex.dev/quickstarts",
93-
children: "Follow a quickstart guide for your favorite framework.",
94-
}}
95-
/>
96-
</Sheet>
57+
<div className="relative h-full w-full">
58+
{/* Background table example */}
59+
<div className="pointer-events-none absolute inset-0 opacity-50">
60+
<div className="flex h-full w-full flex-col">
61+
{/* Example DataToolbar */}
62+
{noTables && (
63+
<div className="mb-2 flex flex-col" inert>
64+
<div className="flex flex-wrap items-end justify-between gap-4">
65+
<div className="flex max-w-full items-center gap-4">
66+
<div className="flex max-w-full flex-col gap-1">
67+
<h3 className="flex select-none items-start gap-0.5 font-mono text-content-secondary">
68+
example_table
69+
</h3>
70+
</div>
71+
</div>
72+
<div className="flex flex-wrap items-center gap-2">
73+
<Button size="sm" variant="neutral" icon={<PlusIcon />}>
74+
Add
75+
</Button>
76+
<Button
77+
size="sm"
78+
variant="neutral"
79+
icon={<DotsVerticalIcon className="m-[3px]" />}
80+
/>
81+
</div>
82+
</div>
83+
</div>
84+
)}
85+
86+
{/* Example DataFilters */}
87+
<div
88+
className="flex w-full flex-col gap-2 rounded-t border border-b-0 bg-background-secondary/50 p-2"
89+
inert
90+
>
91+
<div className="flex justify-between gap-2">
92+
<div className="flex items-center">
93+
<div className="flex w-full rounded bg-background-secondary">
94+
<Button
95+
size="xs"
96+
variant="neutral"
97+
className="w-fit rounded-l-none border border-border-transparent text-xs"
98+
icon={<MixerHorizontalIcon className="size-3.5" />}
99+
>
100+
<div className="flex items-center gap-2">
101+
<span>Filter & Sort</span>
102+
</div>
103+
<ChevronDownIcon />
104+
</Button>
105+
</div>
106+
</div>
107+
<div className="flex gap-2">
108+
<div className="flex items-center gap-1 whitespace-nowrap text-xs">
109+
<span className="h-3 w-24 bg-content-secondary/30" />
110+
</div>
111+
</div>
112+
</div>
113+
</div>
114+
115+
{/* Table */}
116+
<div
117+
className="flex h-full w-full flex-col overflow-hidden rounded rounded-t-none border bg-background-secondary"
118+
inert
119+
>
120+
<table className="h-full w-full table-fixed">
121+
<thead>
122+
<tr className="border-b bg-background-secondary">
123+
{EXAMPLE_COLUMNS.map((col) => (
124+
<th
125+
key={col}
126+
className="select-none border-r p-3 text-left text-xs font-semibold text-content-secondary last:border-r-0"
127+
>
128+
{col}
129+
</th>
130+
))}
131+
</tr>
132+
</thead>
133+
<tbody className="divide-y">
134+
{Array.from({ length: 20 }).map((_, i) => (
135+
<tr key={i} className="group">
136+
{EXAMPLE_COLUMNS.map((col) => (
137+
// eslint-disable-next-line jsx-a11y/control-has-associated-label
138+
<td
139+
key={col}
140+
className="border-r p-3 last:border-r-0 group-last:border-b-0"
141+
>
142+
<div className="h-3 w-full max-w-64 bg-content-secondary/30" />
143+
</td>
144+
))}
145+
</tr>
146+
))}
147+
</tbody>
148+
</table>
149+
</div>
150+
</div>
151+
</div>
152+
153+
{/* Main content */}
154+
<div className="absolute inset-0 flex items-center justify-center">
155+
<Sheet
156+
padding={false}
157+
className="m-6 h-fit w-fit bg-background-secondary/90 p-2 backdrop-blur-[2px]"
158+
>
159+
<EmptySection
160+
Icon={TableIcon}
161+
header={
162+
noTables
163+
? "There are no tables here yet."
164+
: "This table is empty."
165+
}
166+
sheet={false}
167+
body={
168+
noTables
169+
? "Create a table to start storing data."
170+
: "Create a document or run a mutation to start storing data."
171+
}
172+
action={
173+
noTables ? (
174+
<CreateNewTable tableData={tableMetadata} />
175+
) : (
176+
<>
177+
{openAddDocuments && (
178+
<Button
179+
inline
180+
onClick={() => {
181+
log("open add documents panel", { how: "empty data" });
182+
openAddDocuments();
183+
}}
184+
size="sm"
185+
disabled={
186+
!canAddDocuments ||
187+
!!(selectedNent && selectedNent.state !== "active")
188+
}
189+
tip={
190+
selectedNent && selectedNent.state !== "active"
191+
? "Cannot add documents in an unmounted component."
192+
: !canAddDocuments &&
193+
"You do not have permission to add documents in production."
194+
}
195+
icon={<PlusIcon aria-hidden="true" />}
196+
>
197+
Add Documents
198+
</Button>
199+
)}
200+
</>
201+
)
202+
}
203+
learnMoreButton={{
204+
href: "https://docs.convex.dev/quickstarts",
205+
children: "Follow a quickstart guide",
206+
}}
207+
/>
208+
</Sheet>
209+
</div>
210+
</div>
97211
);
98212
}

npm-packages/dashboard-common/src/features/data/components/Table/Table.tsx

+1
Original file line numberDiff line numberDiff line change
@@ -331,6 +331,7 @@ export function Table({
331331
collapsed={collapsed}
332332
direction="left"
333333
panelRef={panelRef}
334+
handleTitle="View Selected"
334335
/>
335336
<Panel
336337
defaultSize={30}

npm-packages/dashboard-common/src/features/data/components/TableTab.tsx

+8-4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import omit from "lodash/omit";
55
import { sidebarLinkClassNames } from "@common/elements/Sidebar";
66
import { Tooltip } from "@common/elements/Tooltip";
77
import { useIsOverflowing } from "@common/lib/useIsOverflowing";
8+
import { cn } from "@common/lib/cn";
89

910
export function TableTab({
1011
selectedTable,
@@ -53,10 +54,13 @@ export function TableTab({
5354
},
5455
}}
5556
key={table}
56-
className={sidebarLinkClassNames({
57-
isActive: selectedTable === table,
58-
small: true,
59-
})}
57+
className={cn(
58+
sidebarLinkClassNames({
59+
isActive: selectedTable === table,
60+
small: true,
61+
}),
62+
"py-1",
63+
)}
6064
onClick={() => onSelectTable?.()}
6165
>
6266
<div className="flex w-full max-w-full items-start gap-0.5">

npm-packages/dashboard-common/src/features/functions/components/DirectorySidebar.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ export function DirectorySidebar({
2929
>
3030
<div className="mb-2 flex flex-col px-3">
3131
<NentSwitcher />
32-
<h5>Function Explorer</h5>
32+
<h5>Functions</h5>
3333
</div>
3434
<input
3535
id="Search functions"

0 commit comments

Comments
 (0)