From 4cf448c35d46c7f80cf65d33246bb134567f4fa8 Mon Sep 17 00:00:00 2001 From: Max Leiter Date: Wed, 16 Nov 2022 02:16:56 -0800 Subject: [PATCH] more geist removal; add popover, convert more of post editing and viewing --- .../file-dropdown/dropdown.module.css | 4 +- .../components/file-dropdown/index.tsx | 85 ++++++------------- .../app/(posts)/components/preview/index.tsx | 1 - client/app/(posts)/components/tabs/index.tsx | 2 +- .../edit-document/document.module.css | 18 ---- .../formatting-icons.module.css | 17 ++++ .../edit-document/formatting-icons/index.tsx | 5 +- .../edit-document/index.tsx | 8 -- .../post/[id]/components/post-page/index.tsx | 29 +++---- .../components/post-page/post-page.module.css | 3 +- .../post-page/view-document/index.tsx | 17 ++-- .../badges/visibility-control/index.tsx | 10 ++- .../button-group/button-group.module.css | 20 +++-- client/app/components/button-group/index.tsx | 13 ++- .../app/components/button/button.module.css | 14 +-- client/app/components/popover/index.tsx | 38 +++++++++ .../app/components/popover/popover.module.css | 25 ++++++ client/app/components/scroll-to-top/index.tsx | 22 ++--- .../scroll-to-top/scroll.module.css | 8 ++ client/app/layout.tsx | 1 - client/package.json | 2 +- client/pnpm-lock.yaml | 4 +- 22 files changed, 191 insertions(+), 155 deletions(-) create mode 100644 client/app/(posts)/new/components/edit-document-list/edit-document/formatting-icons/formatting-icons.module.css create mode 100644 client/app/components/popover/index.tsx create mode 100644 client/app/components/popover/popover.module.css diff --git a/client/app/(posts)/components/file-dropdown/dropdown.module.css b/client/app/(posts)/components/file-dropdown/dropdown.module.css index e113001d..dbb061a7 100644 --- a/client/app/(posts)/components/file-dropdown/dropdown.module.css +++ b/client/app/(posts)/components/file-dropdown/dropdown.module.css @@ -15,7 +15,7 @@ .content { list-style: none; width: 100%; - padding: 0 var(--gap); + padding: var(--gap-quarter) var(--gap); margin: 0; } @@ -26,7 +26,7 @@ padding: 0 0; } -.content li:hover, +.contenthover, .content li:focus { background-color: var(--lighter-gray); } diff --git a/client/app/(posts)/components/file-dropdown/index.tsx b/client/app/(posts)/components/file-dropdown/index.tsx index 5ceb75ab..60aeb0dd 100644 --- a/client/app/(posts)/components/file-dropdown/index.tsx +++ b/client/app/(posts)/components/file-dropdown/index.tsx @@ -1,55 +1,39 @@ +import Button from "@components/button" +import { Popover } from "@components/popover" import ShiftBy from "@components/shift-by" -import { Button, Popover } from "@geist-ui/core/dist" import ChevronDown from "@geist-ui/icons/chevronDown" import CodeIcon from "@geist-ui/icons/fileFunction" import FileIcon from "@geist-ui/icons/fileText" import { codeFileExtensions } from "@lib/constants" +import clsx from "clsx" import type { File } from "lib/server/prisma" -import { useCallback, useEffect, useState } from "react" import styles from "./dropdown.module.css" +import buttonStyles from "@components/button/button.module.css" type Item = File & { icon: JSX.Element } -const FileDropdown = ({ - files, - isMobile -}: { - files: File[] - isMobile: boolean -}) => { - const [expanded, setExpanded] = useState(false) - const [items, setItems] = useState([]) - const changeHandler = (next: boolean) => { - setExpanded(next) - } - - const onOpen = () => setExpanded(true) - const onClose = useCallback(() => setExpanded(false), [setExpanded]) - - useEffect(() => { - const newItems = files.map((file) => { - const extension = file.title.split(".").pop() - if (codeFileExtensions.includes(extension || "")) { - return { - ...file, - icon: - } - } else { - return { - ...file, - icon: - } +const FileDropdown = ({ files }: { files: File[] }) => { + const items = files.map((file) => { + const extension = file.title.split(".").pop() + if (codeFileExtensions.includes(extension || "")) { + return { + ...file, + icon: } - }) - setItems(newItems) - }, [files]) + } else { + return { + ...file, + icon: + } + } + }) const content = ( ) - // a list of files with an icon and a title return ( - <> - - - + + +
+ +
+ Jump to {files.length} {files.length === 1 ? "file" : "files"} +
+ {content} +
) } diff --git a/client/app/(posts)/components/preview/index.tsx b/client/app/(posts)/components/preview/index.tsx index 5fe9b75d..fd668fd2 100644 --- a/client/app/(posts)/components/preview/index.tsx +++ b/client/app/(posts)/components/preview/index.tsx @@ -69,7 +69,6 @@ export const StaticPreview = ({ preview: string height: string | number }) => { - console.log("content", preview) return (
- + {isEditing && }
}) => { diff --git a/client/app/(posts)/new/components/edit-document-list/edit-document/index.tsx b/client/app/(posts)/new/components/edit-document-list/edit-document/index.tsx index debeda94..c27f55a3 100644 --- a/client/app/(posts)/new/components/edit-document-list/edit-document/index.tsx +++ b/client/app/(posts)/new/components/edit-document-list/edit-document/index.tsx @@ -2,17 +2,9 @@ import { ChangeEvent, memo, useCallback, - useMemo, - useRef, - useState } from "react" import styles from "./document.module.css" import Trash from "@geist-ui/icons/trash" -import FormattingIcons from "./formatting-icons" -import TextareaMarkdown, { TextareaMarkdownRef } from "textarea-markdown-editor" - -import { Tabs } from "@geist-ui/core/dist" -import Preview from "../../../../components/preview" import Button from "@components/button" import Input from "@components/input" import DocumentTabs from "app/(posts)/components/tabs" diff --git a/client/app/(posts)/post/[id]/components/post-page/index.tsx b/client/app/(posts)/post/[id]/components/post-page/index.tsx index 2222916c..b2248d92 100644 --- a/client/app/(posts)/post/[id]/components/post-page/index.tsx +++ b/client/app/(posts)/post/[id]/components/post-page/index.tsx @@ -4,7 +4,7 @@ import VisibilityBadge from "@components/badges/visibility-badge" import DocumentComponent from "./view-document" import styles from "./post-page.module.css" -import { Button, ButtonGroup, useMediaQuery } from "@geist-ui/core/dist" +import { useMediaQuery } from "@geist-ui/core/dist" import { useEffect, useState } from "react" import Archive from "@geist-ui/icons/archive" import Edit from "@geist-ui/icons/edit" @@ -17,6 +17,8 @@ import CreatedAgoBadge from "@components/badges/created-ago-badge" import PasswordModalPage from "./password-modal-wrapper" import VisibilityControl from "@components/badges/visibility-control" import { File, PostWithFilesAndAuthor } from "@lib/server/prisma" +import ButtonGroup from "@components/button-group" +import Button from "@components/button" type Props = { post: string | PostWithFilesAndAuthor @@ -99,39 +101,36 @@ const PostPage = ({ post: initialPost, isProtected, isAuthor }: Props) => { {!isAvailable && }
- + {post.parentId && ( - )} - + -

{post.title} by {post.author?.displayName}

+

+ {post.title}{" "} + + by {post.author?.displayName} + +

diff --git a/client/app/(posts)/post/[id]/components/post-page/post-page.module.css b/client/app/(posts)/post/[id]/components/post-page/post-page.module.css index 97d8c94f..21bb10c4 100644 --- a/client/app/(posts)/post/[id]/components/post-page/post-page.module.css +++ b/client/app/(posts)/post/[id]/components/post-page/post-page.module.css @@ -8,6 +8,7 @@ .header .title .badges { display: flex; gap: var(--gap-half); + flex-wrap: wrap; } .header .title h3 { @@ -41,8 +42,8 @@ } .header .title .badges { - flex-direction: column; align-items: center; + justify-content: center; } .header .buttons { diff --git a/client/app/(posts)/post/[id]/components/post-page/view-document/index.tsx b/client/app/(posts)/post/[id]/components/post-page/view-document/index.tsx index 0f293ffb..c7b5bbb4 100644 --- a/client/app/(posts)/post/[id]/components/post-page/view-document/index.tsx +++ b/client/app/(posts)/post/[id]/components/post-page/view-document/index.tsx @@ -1,17 +1,15 @@ -import { memo, useRef } from "react" +import { memo } from "react" import styles from "./document.module.css" import Download from "@geist-ui/icons/download" import ExternalLink from "@geist-ui/icons/externalLink" import Skeleton from "@components/skeleton" import Link from "next/link" -import { Tabs, Textarea, Tag, Spacer } from "@geist-ui/core/dist" -import { StaticPreview } from "app/(posts)/components/preview" -import FadeIn from "@components/fade-in" import Tooltip from "@components/tooltip" import Button from "@components/button" import ButtonGroup from "@components/button-group" import DocumentTabs from "app/(posts)/components/tabs" +import Input from "@components/input" // import Link from "next/link" type Props = { @@ -66,7 +64,6 @@ const Document = ({ if (skeleton) { return ( <> -
@@ -86,14 +83,14 @@ const Document = ({ <>
- - {title || "Untitled"} - + value={title || "Untitled"} + disabled + />
diff --git a/client/app/components/badges/visibility-control/index.tsx b/client/app/components/badges/visibility-control/index.tsx index fca83ebb..af224f47 100644 --- a/client/app/components/badges/visibility-control/index.tsx +++ b/client/app/components/badges/visibility-control/index.tsx @@ -1,6 +1,8 @@ -import { Button, ButtonGroup, Loading, useToasts } from "@geist-ui/core/dist" +import { Loading, useToasts } from "@geist-ui/core/dist" import PasswordModal from "@components/password-modal" import { useCallback, useState } from "react" +import ButtonGroup from "@components/button-group" +import Button from "@components/button" type Props = { postId: string @@ -65,12 +67,12 @@ const VisibilityControl = ({ postId, visibility, setVisibility }: Props) => { {isSubmitting ? ( ) : ( - + )} diff --git a/client/app/components/button-group/button-group.module.css b/client/app/components/button-group/button-group.module.css index c90088f0..89983fdd 100644 --- a/client/app/components/button-group/button-group.module.css +++ b/client/app/components/button-group/button-group.module.css @@ -3,8 +3,6 @@ flex-direction: row; flex-wrap: nowrap; justify-content: flex-start; - align-items: stretch; - align-content: stretch; } .button-group > * { @@ -12,8 +10,20 @@ margin: 0; } -/* all buttons on the inside should have no border radius */ - -.button-group :global(button) { +.button-group > * { border-radius: 0 !important; } + +.button-group > button:first-of-type { + border-top-left-radius: var(--radius) !important; + border-bottom-left-radius: var(--radius) !important; +} + +.button-group > button:last-of-type { + border-top-right-radius: var(--radius) !important; + border-bottom-right-radius: var(--radius) !important; +} + +.vertical { + flex-direction: column; +} diff --git a/client/app/components/button-group/index.tsx b/client/app/components/button-group/index.tsx index a0713bc3..01e5fb9d 100644 --- a/client/app/components/button-group/index.tsx +++ b/client/app/components/button-group/index.tsx @@ -1,13 +1,22 @@ import styles from "./button-group.module.css" - +import clsx from "clsx" export default function ButtonGroup({ children, + vertical, ...props }: { children: React.ReactNode | React.ReactNode[] + vertical?: boolean } & React.HTMLAttributes) { return ( -
+
{children}
) diff --git a/client/app/components/button/button.module.css b/client/app/components/button/button.module.css index 05dbe050..047b8b38 100644 --- a/client/app/components/button/button.module.css +++ b/client/app/components/button/button.module.css @@ -1,8 +1,3 @@ -.button:root { - --hover: var(--bg); - --hover-bg: var(--fg); -} - .button { user-select: none; cursor: pointer; @@ -14,8 +9,8 @@ .button:hover, .button:focus { - color: var(--hover); - background: var(--hover-bg); + color: var(--fg); + background: var(--bg); border: 1px solid var(--darker-gray); } @@ -25,6 +20,11 @@ color: var(--gray); } +.button[disabled]:hover, +.button[disabled]:focus { + border: 1px solid currentColor; +} + .secondary { background: var(--bg); color: var(--fg); diff --git a/client/app/components/popover/index.tsx b/client/app/components/popover/index.tsx new file mode 100644 index 00000000..8edcc95a --- /dev/null +++ b/client/app/components/popover/index.tsx @@ -0,0 +1,38 @@ +// largely from https://github.com/shadcn/taxonomy + +import * as React from "react" +import * as PopoverPrimitive from "@radix-ui/react-popover" +import clsx from "clsx" +import styles from './popover.module.css' + +type PopoverProps = PopoverPrimitive.PopoverProps + +export function Popover({ ...props }: PopoverProps) { + return +} + +Popover.Trigger = React.forwardRef< + HTMLButtonElement, + PopoverPrimitive.PopoverTriggerProps +>(function PopoverTrigger({ ...props }, ref) { + return +}) + +Popover.Portal = PopoverPrimitive.Portal + +Popover.Content = React.forwardRef< + HTMLDivElement, + PopoverPrimitive.PopoverContentProps +>(function PopoverContent({ className, ...props }, ref) { + return ( + + ) +}) diff --git a/client/app/components/popover/popover.module.css b/client/app/components/popover/popover.module.css new file mode 100644 index 00000000..318a4955 --- /dev/null +++ b/client/app/components/popover/popover.module.css @@ -0,0 +1,25 @@ +.root { + overflow: hidden; + border-radius: 0.375rem; + border: 1px solid var(--border); + background-color: var(--bg); + box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06); + animation: slide-in-from-top 0.1s cubic-bezier(0.4, 0, 1, 1) 0.1s; + animation-fill-mode: both; + width: 100%; +} + +@keyframes slide-in-from-top { + 0% { + transform: translateY(-10px); + opacity: 0; + } + 100% { + transform: translateY(0); + opacity: 1; + } +} + + + + diff --git a/client/app/components/scroll-to-top/index.tsx b/client/app/components/scroll-to-top/index.tsx index 29f416ab..20262081 100644 --- a/client/app/components/scroll-to-top/index.tsx +++ b/client/app/components/scroll-to-top/index.tsx @@ -1,6 +1,6 @@ +import Button from "@components/button" import Tooltip from "@components/tooltip" -import { Button, Spacer } from "@geist-ui/core/dist" -import ChevronUp from "@geist-ui/icons/chevronUpCircleFill" +import ChevronUp from "@geist-ui/icons/chevronUp" import { useEffect, useState } from "react" import styles from "./scroll.module.css" @@ -26,15 +26,7 @@ const ScrollToTop = () => { } return ( -
+
{ + iconLeft={} + />
) diff --git a/client/app/components/scroll-to-top/scroll.module.css b/client/app/components/scroll-to-top/scroll.module.css index 070e2cae..a5ca4b3f 100644 --- a/client/app/components/scroll-to-top/scroll.module.css +++ b/client/app/components/scroll-to-top/scroll.module.css @@ -1,3 +1,11 @@ +.root { + display: flex; + flex-direction: row; + width: 100%; + height: 24px; + justify-content: flex-end; +} + .scroll-up { position: fixed; z-index: 2; diff --git a/client/app/layout.tsx b/client/app/layout.tsx index ca3d0881..2b98bd66 100644 --- a/client/app/layout.tsx +++ b/client/app/layout.tsx @@ -25,7 +25,6 @@ export default async function RootLayout({ children }: RootLayoutProps) { .find(row => row.startsWith('${THEME_COOKIE_NAME}=')) .split('=')[1]; document.documentElement.setAttribute('data-theme', theme); - console.log("theme on load", theme) })(); `, }} diff --git a/client/package.json b/client/package.json index 5c77cdb0..c794a064 100644 --- a/client/package.json +++ b/client/package.json @@ -23,7 +23,6 @@ "@wcj/markdown-to-html": "^2.1.2", "bcrypt": "^5.1.0", "client-zip": "2.2.1", - "clsx": "^1.2.1", "cookies-next": "^2.1.1", "jest": "^29.3.1", "next": "13.0.3", @@ -46,6 +45,7 @@ "@types/react": "18.0.9", "@types/react-datepicker": "4.4.1", "@types/react-dom": "18.0.3", + "clsx": "^1.2.1", "cross-env": "7.0.3", "eslint": "8.27.0", "eslint-config-next": "13.0.3", diff --git a/client/pnpm-lock.yaml b/client/pnpm-lock.yaml index 4f03b100..7e698b9e 100644 --- a/client/pnpm-lock.yaml +++ b/client/pnpm-lock.yaml @@ -54,7 +54,6 @@ dependencies: '@wcj/markdown-to-html': 2.1.2 bcrypt: 5.1.0 client-zip: 2.2.1 - clsx: 1.2.1 cookies-next: 2.1.1 jest: 29.3.1_@types+node@17.0.23 next: 13.0.3_biqbaboplfbrettd7655fr4n2y @@ -80,6 +79,7 @@ devDependencies: '@types/react': 18.0.9 '@types/react-datepicker': 4.4.1_biqbaboplfbrettd7655fr4n2y '@types/react-dom': 18.0.3 + clsx: 1.2.1 cross-env: 7.0.3 eslint: 8.27.0 eslint-config-next: 13.0.3_hsmo2rtalirsvadpuxki35bq2i @@ -2159,7 +2159,7 @@ packages: /clsx/1.2.1: resolution: {integrity: sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==} engines: {node: '>=6'} - dev: false + dev: true /co/4.6.0: resolution: {integrity: sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==}