Batch of fixes from the automated QA pass on develop. Each was reproduced and then verified fixed live (browser/curl); logic-bearing fixes have unit tests. Functional bugs: - #122 collab-token was capped by the anonymous public-share-AI throttler (5/min); skip all non-AUTH named throttlers on this auth-guarded, client-cached route. - #123 editor onAuthenticationFailed threw `jwtDecode(undefined)` and never reconnected; read the token via a ref, guard the decode (incl. missing exp), and refetch+reconnect on any auth failure. - #124 a slash command containing a space ("/Heading 1") inserted literal text; enable allowSpaces and close the menu when the query matches no items. - #125 space slug auto-gen produced uppercase initials for multi-word names; computeSpaceSlug now yields a lowercase alphanumeric slug. - #126 AI chat window position/size now persisted (atomWithStorage) across reload; also fixes a latent ResizeObserver-attach bug on first open. - #127 workspace name update accepted URLs; add @NoUrls (parity with setup). - #132 icon-columns 4/5 passed calc() into SVG width/height attrs (console spam); size via style. share-for-page query returns null instead of undefined. - #134 "Reindex now" counter looked stuck: reindex runs async; the client now polls coverage (bounded) so the counter climbs live; misleading server comment reworded. UX / consistency: - #128 add success toasts to favorite/label/avatar/member-(de)activate. - #129 "1 result found" pluralization; hide the single-option Type filter. - #130 replace raw Zod strings with friendly messages (name/password/group). - #131 unify "Untitled" casing in tree/breadcrumb/tab; stop force-uppercasing space-name chips; fix confirm-dialog labels (Cancel / Remove), invite placeholder typo, Export/Move-to-space labels. - #133 disable profile Save when clean; toast on unsupported avatar image; style the invalid-invitation page with a CTA; hide Share for read-only users; align the dictation "not configured" message; "Go to login page" typo. Tests: computeSpaceSlug, workspace-name NoUrls DTO, share-query null normalization, slash getSuggestionItems empty-close. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
115 lines
3.5 KiB
TypeScript
115 lines
3.5 KiB
TypeScript
import {
|
|
Text,
|
|
Group,
|
|
UnstyledButton,
|
|
Badge,
|
|
Table,
|
|
ThemeIcon,
|
|
Button,
|
|
} from "@mantine/core";
|
|
import { Link } from "react-router-dom";
|
|
import PageListSkeleton from "@/components/ui/page-list-skeleton.tsx";
|
|
import { buildPageUrl } from "@/features/page/page.utils.ts";
|
|
import { formattedDate } from "@/lib/time.ts";
|
|
import { useRecentChangesQuery } from "@/features/page/queries/page-query.ts";
|
|
import { IconFileDescription, IconFiles } from "@tabler/icons-react";
|
|
import { EmptyState } from "@/components/ui/empty-state.tsx";
|
|
import { getSpaceUrl } from "@/lib/config.ts";
|
|
import { useTranslation } from "react-i18next";
|
|
import { getInitialsColor } from "@/lib/get-initials-color.ts";
|
|
import rowClasses from "@/components/ui/clickable-table-row.module.css";
|
|
|
|
interface Props {
|
|
spaceId?: string;
|
|
}
|
|
|
|
export default function RecentChanges({ spaceId }: Props) {
|
|
const { t } = useTranslation();
|
|
const { data, isLoading, isError, hasNextPage, fetchNextPage, isFetchingNextPage } = useRecentChangesQuery(spaceId);
|
|
const pages = data?.pages.flatMap((p) => p.items) ?? [];
|
|
|
|
if (isLoading) {
|
|
return <PageListSkeleton />;
|
|
}
|
|
|
|
if (isError) {
|
|
return <Text>{t("Failed to fetch recent pages")}</Text>;
|
|
}
|
|
|
|
return pages.length > 0 ? (
|
|
<>
|
|
<Table.ScrollContainer minWidth={500}>
|
|
<Table highlightOnHover verticalSpacing={6}>
|
|
<Table.Tbody>
|
|
{pages.map((page) => (
|
|
<Table.Tr key={page.id} className={rowClasses.row}>
|
|
<Table.Td>
|
|
<UnstyledButton
|
|
className={rowClasses.link}
|
|
component={Link}
|
|
to={buildPageUrl(page?.space.slug, page.slugId, page.title)}
|
|
>
|
|
<Group wrap="nowrap">
|
|
{page.icon || (
|
|
<ThemeIcon variant="transparent" color="gray" size={18}>
|
|
<IconFileDescription size={18} />
|
|
</ThemeIcon>
|
|
)}
|
|
|
|
<Text fw={500} size="md" lineClamp={1}>
|
|
{page.title || t("Untitled")}
|
|
</Text>
|
|
</Group>
|
|
</UnstyledButton>
|
|
</Table.Td>
|
|
{!spaceId && (
|
|
<Table.Td>
|
|
<Badge
|
|
color={getInitialsColor(page?.space.name)}
|
|
variant="light"
|
|
tt="none"
|
|
component={Link}
|
|
to={getSpaceUrl(page?.space.slug)}
|
|
style={{ cursor: "pointer" }}
|
|
>
|
|
{page?.space.name}
|
|
</Badge>
|
|
</Table.Td>
|
|
)}
|
|
<Table.Td>
|
|
<Text
|
|
c="dimmed"
|
|
style={{ whiteSpace: "nowrap" }}
|
|
size="xs"
|
|
fw={500}
|
|
>
|
|
{formattedDate(page.updatedAt)}
|
|
</Text>
|
|
</Table.Td>
|
|
</Table.Tr>
|
|
))}
|
|
</Table.Tbody>
|
|
</Table>
|
|
</Table.ScrollContainer>
|
|
{hasNextPage && (
|
|
<Button
|
|
variant="subtle"
|
|
fullWidth
|
|
mt="sm"
|
|
mb="xl"
|
|
onClick={() => fetchNextPage()}
|
|
loading={isFetchingNextPage}
|
|
>
|
|
{t("Load more")}
|
|
</Button>
|
|
)}
|
|
</>
|
|
) : (
|
|
<EmptyState
|
|
icon={IconFiles}
|
|
title={t("No pages yet")}
|
|
description={t("Pages you create will show up here.")}
|
|
/>
|
|
);
|
|
}
|