remove more of geist-ui: add spinner, button dropdown, toasts. bump deps

This commit is contained in:
Max Leiter 2022-11-28 18:33:06 -08:00
parent 0cab3acd62
commit d6894ffb8b
35 changed files with 397 additions and 459 deletions

View file

@ -60,7 +60,7 @@
} }
.chevron { .chevron {
transition: transform 0.2s ease-in-out; transition: transform 0.1s ease-in-out;
} }
[data-state="open"] .chevron { [data-state="open"] .chevron {

View file

@ -2,7 +2,8 @@ import { memo, useEffect, useState } from "react"
import styles from "./preview.module.css" import styles from "./preview.module.css"
import "@styles/markdown.css" import "@styles/markdown.css"
import "@styles/syntax.css" import "@styles/syntax.css"
import { Spinner } from "@geist-ui/core/dist" import Skeleton from "@components/skeleton"
import { Spinner } from "@components/spinner"
type Props = { type Props = {
height?: number | string height?: number | string
@ -52,9 +53,7 @@ const MarkdownPreview = ({
return ( return (
<> <>
{isLoading ? ( {isLoading ? (
<>
<Spinner /> <Spinner />
</>
) : ( ) : (
<StaticPreview preview={preview} height={height} /> <StaticPreview preview={preview} height={height} />
)} )}

View file

@ -1,3 +1,5 @@
"use client"
import * as RadixTabs from "@radix-ui/react-tabs" import * as RadixTabs from "@radix-ui/react-tabs"
import FormattingIcons from "app/(posts)/new/components/edit-document-list/edit-document/formatting-icons" import FormattingIcons from "app/(posts)/new/components/edit-document-list/edit-document/formatting-icons"
import { ChangeEvent, useRef } from "react" import { ChangeEvent, useRef } from "react"

View file

@ -17,7 +17,7 @@
border-radius: 2px; border-radius: 2px;
border: 2px dashed var(--border) !important; border: 2px dashed var(--border) !important;
outline: none; outline: none;
transition: all 0.24s ease-in-out; transition: all 0.14s ease-in-out;
cursor: pointer; cursor: pointer;
} }
@ -32,7 +32,7 @@
.error { .error {
color: red; color: red;
font-size: 0.8rem; font-size: 0.8rem;
transition: border 0.24s ease-in-out; transition: border 0.14s ease-in-out;
border: 2px solid red; border: 2px solid red;
border-radius: 2px; border-radius: 2px;
padding: 20px; padding: 20px;
@ -42,3 +42,13 @@
margin: 0; margin: 0;
padding-left: var(--gap-double); padding-left: var(--gap-double);
} }
.verb:after {
content: "click";
}
@media (hover: none) {
.verb:after {
content: "tap";
}
}

View file

@ -1,4 +1,3 @@
import { useMediaQuery, useTheme, useToasts } from "@geist-ui/core/dist"
import { useDropzone } from "react-dropzone" import { useDropzone } from "react-dropzone"
import styles from "./drag-and-drop.module.css" import styles from "./drag-and-drop.module.css"
import generateUUID from "@lib/generate-uuid" import generateUUID from "@lib/generate-uuid"
@ -9,13 +8,10 @@ import {
} from "@lib/constants" } from "@lib/constants"
import byteToMB from "@lib/byte-to-mb" import byteToMB from "@lib/byte-to-mb"
import type { Document } from "../new" import type { Document } from "../new"
import { useToasts } from "@components/toasts"
function FileDropzone({ setDocs }: { setDocs: (docs: Document[]) => void }) { function FileDropzone({ setDocs }: { setDocs: (docs: Document[]) => void }) {
const { palette } = useTheme()
const { setToast } = useToasts() const { setToast } = useToasts()
const isMobile = useMediaQuery("xs", {
match: "down"
})
const onDrop = async (acceptedFiles: File[]) => { const onDrop = async (acceptedFiles: File[]) => {
const newDocs = await Promise.all( const newDocs = await Promise.all(
acceptedFiles.map((file) => { acceptedFiles.map((file) => {
@ -23,9 +19,9 @@ function FileDropzone({ setDocs }: { setDocs: (docs: Document[]) => void }) {
const reader = new FileReader() const reader = new FileReader()
reader.onabort = () => reader.onabort = () =>
setToast({ text: "File reading was aborted", type: "error" }) setToast({ message: "File reading was aborted", type: "error" })
reader.onerror = () => reader.onerror = () =>
setToast({ text: "File reading failed", type: "error" }) setToast({ message: "File reading failed", type: "error" })
reader.onload = () => { reader.onload = () => {
const content = reader.result as string const content = reader.result as string
resolve({ resolve({
@ -84,20 +80,13 @@ function FileDropzone({ setDocs }: { setDocs: (docs: Document[]) => void }) {
</li> </li>
)) ))
const verb = isMobile ? "tap" : "click"
return ( return (
<div className={styles.container}> <div className={styles.container}>
<div <div {...getRootProps()} className={styles.dropzone}>
{...getRootProps()}
className={styles.dropzone}
style={{
borderColor: palette.accents_3
}}
>
<input {...getInputProps()} /> <input {...getInputProps()} />
{!isDragActive && ( {!isDragActive && (
<p style={{ color: "var(--gray)" }}> <p style={{ color: "var(--gray)" }}>
Drag some files here, or {verb} to select files Drag some files here, or <span className={styles.verb} /> to select files
</p> </p>
)} )}
{isDragActive && <p>Release to drop the files here</p>} {isDragActive && <p>Release to drop the files here</p>}

View file

@ -1,7 +1,7 @@
.card { .card {
margin: var(--gap) auto; margin: var(--gap) auto;
padding: var(--gap); padding: var(--gap);
border: 1px solid var(--light-gray); border: 1px solid var(--lighter-gray);
border-radius: var(--radius); border-radius: var(--radius);
} }

View file

@ -6,7 +6,7 @@
.actionWrapper .actions { .actionWrapper .actions {
position: absolute; position: absolute;
right: 0; right: 0;
top: -34px; top: -40px;
} }
/* small screens, top: 0 */ /* small screens, top: 0 */

View file

@ -1,6 +1,5 @@
"use client" "use client"
import { useToasts, ButtonDropdown } from "@geist-ui/core/dist"
import { useRouter } from "next/navigation" import { useRouter } from "next/navigation"
import { useCallback, useState } from "react" import { useCallback, useState } from "react"
import generateUUID from "@lib/generate-uuid" import generateUUID from "@lib/generate-uuid"
@ -16,6 +15,8 @@ import Title from "./title"
import FileDropzone from "./drag-and-drop" import FileDropzone from "./drag-and-drop"
import Button from "@components/button" import Button from "@components/button"
import Input from "@components/input" import Input from "@components/input"
import ButtonDropdown from "@components/button-dropdown"
import { useToasts } from "@components/toasts"
const emptyDoc = { const emptyDoc = {
title: "", title: "",
content: "", content: "",
@ -87,7 +88,8 @@ const Post = ({
const json = await res.json() const json = await res.json()
console.error(json) console.error(json)
setToast({ setToast({
text: "Please fill out all fields", id: "error",
message: "Please fill out all fields",
type: "error" type: "error"
}) })
setPasswordModalVisible(false) setPasswordModalVisible(false)
@ -114,7 +116,7 @@ const Post = ({
if (!title) { if (!title) {
setToast({ setToast({
text: "Please fill out the post title", message: "Please fill out the post title",
type: "error" type: "error"
}) })
hasErrored = true hasErrored = true
@ -122,7 +124,7 @@ const Post = ({
if (!docs.length) { if (!docs.length) {
setToast({ setToast({
text: "Please add at least one document", message: "Please add at least one document",
type: "error" type: "error"
}) })
hasErrored = true hasErrored = true
@ -131,7 +133,7 @@ const Post = ({
for (const doc of docs) { for (const doc of docs) {
if (!doc.title) { if (!doc.title) {
setToast({ setToast({
text: "Please fill out all the document titles", message: "Please fill out all the document titles",
type: "error" type: "error"
}) })
hasErrored = true hasErrored = true
@ -308,19 +310,27 @@ const Post = ({
enableTabLoop={false} enableTabLoop={false}
minDate={new Date()} minDate={new Date()}
/> />
<ButtonDropdown loading={isSubmitting} type="success"> <ButtonDropdown iconHeight={40}>
<ButtonDropdown.Item main onClick={() => onSubmit("unlisted")}> <Button
height={40}
width={251}
onClick={() => onSubmit("unlisted")}
>
Create Unlisted Create Unlisted
</ButtonDropdown.Item> </Button>
<ButtonDropdown.Item onClick={() => onSubmit("private")}> <Button height={40} width={300} onClick={() => onSubmit("private")}>
Create Private Create Private
</ButtonDropdown.Item> </Button>
<ButtonDropdown.Item onClick={() => onSubmit("public")}> <Button height={40} width={300} onClick={() => onSubmit("public")}>
Create Public Create Public
</ButtonDropdown.Item> </Button>
<ButtonDropdown.Item onClick={() => onSubmit("protected")}> <Button
height={40}
width={300}
onClick={() => onSubmit("protected")}
>
Create with Password Create with Password
</ButtonDropdown.Item> </Button>
</ButtonDropdown> </ButtonDropdown>
</div> </div>
</div> </div>

View file

@ -13,14 +13,14 @@ const titlePlaceholders = [
"I'm thinking about ..." "I'm thinking about ..."
] ]
const placeholder = titlePlaceholders[3]
type props = { type props = {
onChange: (e: ChangeEvent<HTMLInputElement>) => void onChange: (e: ChangeEvent<HTMLInputElement>) => void
title?: string title?: string
} }
const Title = ({ onChange, title }: props) => { const Title = ({ onChange, title }: props) => {
const placeholder =
titlePlaceholders[Math.floor(Math.random() * titlePlaceholders.length)]
return ( return (
<div className={styles.title}> <div className={styles.title}>
<h1>Drift</h1> <h1>Drift</h1>

View file

@ -4,7 +4,6 @@ import VisibilityBadge from "@components/badges/visibility-badge"
import DocumentComponent from "./view-document" import DocumentComponent from "./view-document"
import styles from "./post-page.module.css" import styles from "./post-page.module.css"
import { useMediaQuery } from "@geist-ui/core/dist"
import { useEffect, useState } from "react" import { useEffect, useState } from "react"
import Archive from "@geist-ui/icons/archive" import Archive from "@geist-ui/icons/archive"
import Edit from "@geist-ui/icons/edit" import Edit from "@geist-ui/icons/edit"
@ -32,7 +31,6 @@ const PostPage = ({ post: initialPost, isProtected, isAuthor }: Props) => {
) )
const [visibility, setVisibility] = useState<string>(post.visibility) const [visibility, setVisibility] = useState<string>(post.visibility)
const router = useRouter() const router = useRouter()
const isMobile = useMediaQuery("mobile")
useEffect(() => { useEffect(() => {
if (post.expiresAt) { if (post.expiresAt) {
@ -101,7 +99,7 @@ const PostPage = ({ post: initialPost, isProtected, isAuthor }: Props) => {
{!isAvailable && <PasswordModalPage setPost={setPost} postId={post.id} />} {!isAvailable && <PasswordModalPage setPost={setPost} postId={post.id} />}
<div className={styles.header}> <div className={styles.header}>
<span className={styles.buttons}> <span className={styles.buttons}>
<ButtonGroup vertical={isMobile}> <ButtonGroup verticalIfMobile>
<Button <Button
iconLeft={<Edit />} iconLeft={<Edit />}
onClick={editACopy} onClick={editACopy}

View file

@ -1,8 +1,8 @@
import { useToasts } from "@geist-ui/core/dist"
import { Post, PostWithFilesAndAuthor } from "@lib/server/prisma" import { Post, PostWithFilesAndAuthor } from "@lib/server/prisma"
import PasswordModal from "@components/password-modal" import PasswordModal from "@components/password-modal"
import { useRouter } from "next/navigation" import { useRouter } from "next/navigation"
import { useState } from "react" import { useState } from "react"
import { useToasts } from "@components/toasts"
type Props = { type Props = {
setPost: (post: PostWithFilesAndAuthor) => void setPost: (post: PostWithFilesAndAuthor) => void
@ -25,7 +25,7 @@ const PasswordModalPage = ({ setPost, postId }: Props) => {
if (!res.ok) { if (!res.ok) {
setToast({ setToast({
type: "error", type: "error",
text: "Wrong password" message: "Wrong password"
}) })
return return
} }
@ -34,7 +34,7 @@ const PasswordModalPage = ({ setPost, postId }: Props) => {
if (data) { if (data) {
if (data.error) { if (data.error) {
setToast({ setToast({
text: data.error, message: data.error,
type: "error" type: "error"
}) })
} else { } else {

View file

@ -35,7 +35,7 @@
} }
} }
@media screen and (max-width: 700px) { @media screen and (max-width: 768px) {
.header .title { .header .title {
flex-direction: column; flex-direction: column;
gap: var(--gap-half); gap: var(--gap-half);
@ -50,4 +50,8 @@
display: flex; display: flex;
justify-content: center; justify-content: center;
} }
.controls {
justify-content: center;
}
} }

View file

@ -1,6 +1,6 @@
"use client" "use client"
import SettingsGroup from "@components/settings-group" import SettingsGroup from "@components/settings-group"
import { Fieldset, useToasts } from "@geist-ui/core/dist" import { Fieldset } from "@geist-ui/core/dist"
import byteToMB from "@lib/byte-to-mb" import byteToMB from "@lib/byte-to-mb"
import { PostWithFiles } from "@lib/server/prisma" import { PostWithFiles } from "@lib/server/prisma"
import Table from "rc-table" import Table from "rc-table"

View file

@ -1,10 +1,11 @@
"use client" "use client"
import { Fieldset, useToasts } from "@geist-ui/core/dist" import { Fieldset } from "@geist-ui/core/dist"
import Table from "rc-table" import Table from "rc-table"
import ActionDropdown from "./action-dropdown" import ActionDropdown from "./action-dropdown"
import SettingsGroup from "@components/settings-group" import SettingsGroup from "@components/settings-group"
import type { User, UserWithPosts } from "@lib/server/prisma" import type { User, UserWithPosts } from "@lib/server/prisma"
import { useState } from "react" import { useState } from "react"
import { useToasts } from "@components/toasts"
const UserTable = ({ users: initial }: { users: UserWithPosts[] }) => { const UserTable = ({ users: initial }: { users: UserWithPosts[] }) => {
const [users, setUsers] = useState(initial) const [users, setUsers] = useState(initial)
@ -21,7 +22,7 @@ const UserTable = ({ users: initial }: { users: UserWithPosts[] }) => {
if (res.status === 200) { if (res.status === 200) {
setToast({ setToast({
text: "Role updated", message: "Role updated",
type: "success" type: "success"
}) })
@ -39,7 +40,7 @@ const UserTable = ({ users: initial }: { users: UserWithPosts[] }) => {
}) })
} else { } else {
setToast({ setToast({
text: "Something went wrong", message: "Something went wrong",
type: "error" type: "error"
}) })
} }

View file

@ -1,8 +1,9 @@
import { Loading, useToasts } from "@geist-ui/core/dist"
import PasswordModal from "@components/password-modal" import PasswordModal from "@components/password-modal"
import { useCallback, useState } from "react" import { useCallback, useState } from "react"
import ButtonGroup from "@components/button-group" import ButtonGroup from "@components/button-group"
import Button from "@components/button" import Button from "@components/button"
import { useToasts } from "@components/toasts"
import { Spinner } from "@components/spinner"
type Props = { type Props = {
postId: string postId: string
@ -30,7 +31,7 @@ const VisibilityControl = ({ postId, visibility, setVisibility }: Props) => {
setVisibility(json.visibility) setVisibility(json.visibility)
} else { } else {
setToast({ setToast({
text: "An error occurred", message: "An error occurred",
type: "error" type: "error"
}) })
setPasswordModalVisible(false) setPasswordModalVisible(false)
@ -65,9 +66,9 @@ const VisibilityControl = ({ postId, visibility, setVisibility }: Props) => {
return ( return (
<> <>
{isSubmitting ? ( {isSubmitting ? (
<Loading /> <Spinner />
) : ( ) : (
<ButtonGroup> <ButtonGroup verticalIfMobile>
<Button <Button
disabled={visibility === "private"} disabled={visibility === "private"}
onClick={() => onSubmit("private")} onClick={() => onSubmit("private")}

View file

@ -1,22 +1,23 @@
.main {
margin-bottom: 2rem;
}
.dropdown { .dropdown {
position: relative; display: flex;
display: inline-block; align-items: stretch;
vertical-align: middle;
cursor: pointer;
padding: 0;
border: 0;
background: transparent;
} }
.dropdownContent { .dropdown > button:first-child {
background-clip: padding-box; border-top-right-radius: 0;
border: 1px solid rgba(0, 0, 0, 0.15); border-bottom-right-radius: 0;
border-radius: 0.25rem; }
box-shadow: 0 3px 12px rgba(0, 0, 0, 0.15);
.dropdown > button:last-child {
border-top-left-radius: 0;
border-bottom-left-radius: 0;
/* if we draw bother borders it will be 2 thick */
border-left: 0;
}
.dropdown > button:last-child:hover {
/* because we removed the left border we need to mock it */
outline: 1px solid var(--darker-gray);
} }
.icon { .icon {

View file

@ -3,6 +3,7 @@ import React, { useCallback, useEffect } from "react"
import { useState } from "react" import { useState } from "react"
import styles from "./dropdown.module.css" import styles from "./dropdown.module.css"
import DownIcon from "@geist-ui/icons/arrowDown" import DownIcon from "@geist-ui/icons/arrowDown"
import * as DropdownMenu from "@radix-ui/react-dropdown-menu"
type Props = { type Props = {
type?: "primary" | "secondary" type?: "primary" | "secondary"
loading?: boolean loading?: boolean
@ -17,99 +18,37 @@ type ButtonDropdownProps = Props & Attrs
const ButtonDropdown: React.FC< const ButtonDropdown: React.FC<
React.PropsWithChildren<ButtonDropdownProps> React.PropsWithChildren<ButtonDropdownProps>
> = ({ type, className, disabled, loading, iconHeight = 24, ...props }) => { > = ({ type, className, disabled, loading, iconHeight = 24, ...props }) => {
const [visible, setVisible] = useState(false)
const [dropdown, setDropdown] = useState<HTMLDivElement | null>(null)
const onClick = (e: React.MouseEvent<HTMLButtonElement>) => {
e.stopPropagation()
e.nativeEvent.stopImmediatePropagation()
setVisible(!visible)
}
const onBlur = () => {
setVisible(false)
}
const onMouseDown = (e: React.MouseEvent<HTMLDivElement>) => {
e.stopPropagation()
e.nativeEvent.stopImmediatePropagation()
}
const onMouseUp = (e: React.MouseEvent<HTMLDivElement>) => {
e.stopPropagation()
e.nativeEvent.stopImmediatePropagation()
}
const onMouseLeave = (e: React.MouseEvent<HTMLDivElement>) => {
e.stopPropagation()
e.nativeEvent.stopImmediatePropagation()
setVisible(false)
}
const onKeyDown = (e: React.KeyboardEvent<HTMLDivElement>) => {
if (e.key === "Escape") {
setVisible(false)
}
}
const onClickOutside = useCallback(
() => (e: React.MouseEvent<HTMLDivElement>) => {
if (dropdown && !dropdown.contains(e.target as Node)) {
setVisible(false)
}
},
[dropdown]
)
useEffect(() => {
if (visible) {
document.addEventListener("mousedown", onClickOutside)
} else {
document.removeEventListener("mousedown", onClickOutside)
}
return () => {
document.removeEventListener("mousedown", onClickOutside)
}
}, [visible, onClickOutside])
if (!Array.isArray(props.children)) { if (!Array.isArray(props.children)) {
return null return null
} }
return ( return (
<div <DropdownMenu.Root>
className={`${styles.main} ${className || ""}`} <div className={styles.dropdown}>
onMouseDown={onMouseDown} {props.children[0]}
onMouseUp={onMouseUp} <DropdownMenu.Trigger
onMouseLeave={onMouseLeave}
onKeyDown={onKeyDown}
onBlur={onBlur}
>
<div
style={{ style={{
display: "flex", display: "flex",
flexDirection: "row", flexDirection: "row",
justifyContent: "flex-end" justifyContent: "flex-end"
}} }}
asChild
> >
{props.children[0]}
<Button <Button
style={{ height: iconHeight, width: iconHeight }} iconLeft={<DownIcon />}
type={type}
className={styles.icon} className={styles.icon}
onClick={() => setVisible(!visible)} />
> </DropdownMenu.Trigger>
<DownIcon /> <DropdownMenu.Portal>
</Button> <DropdownMenu.Content align="end">
</div> {props.children.slice(1).map((child, index) => (
{visible && ( <DropdownMenu.Item key={index}>{child}</DropdownMenu.Item>
<div className={`${styles.dropdown}`}> ))}
<div className={`${styles.dropdownContent}`}> </DropdownMenu.Content>
{props.children.slice(1)} </DropdownMenu.Portal>
</div>
</div>
)}
</div> </div>
</DropdownMenu.Root>
) )
} }

View file

@ -24,6 +24,22 @@
border-bottom-right-radius: var(--radius) !important; border-bottom-right-radius: var(--radius) !important;
} }
.vertical { @media screen and (max-width: 768px) {
.verticalIfMobile {
flex-direction: column; flex-direction: column;
} }
.verticalIfMobile.button-group > button:first-of-type {
border-top-left-radius: var(--radius) !important;
border-top-right-radius: var(--radius) !important;
border-bottom-left-radius: 0 !important;
border-bottom-right-radius: 0 !important;
}
.verticalIfMobile.button-group > button:last-of-type {
border-top-left-radius: 0 !important;
border-top-right-radius: 0 !important;
border-bottom-left-radius: var(--radius) !important;
border-bottom-right-radius: var(--radius) !important;
}
}

View file

@ -2,18 +2,18 @@ import styles from "./button-group.module.css"
import clsx from "clsx" import clsx from "clsx"
export default function ButtonGroup({ export default function ButtonGroup({
children, children,
vertical, verticalIfMobile,
...props ...props
}: { }: {
children: React.ReactNode | React.ReactNode[] children: React.ReactNode | React.ReactNode[]
vertical?: boolean verticalIfMobile?: boolean
} & React.HTMLAttributes<HTMLDivElement>) { } & React.HTMLAttributes<HTMLDivElement>) {
return ( return (
<div <div
className={clsx( className={clsx(
props.className, props.className,
styles["button-group"], styles["button-group"],
vertical && styles.vertical verticalIfMobile && styles.verticalIfMobile
)} )}
{...props} {...props}
> >

View file

@ -1,36 +0,0 @@
import React, { useEffect, useState } from "react"
import MoonIcon from "@geist-ui/icons/moon"
import SunIcon from "@geist-ui/icons/sun"
import styles from "./header.module.css"
import { Select } from "@geist-ui/core/dist"
import { useTheme } from "@components/theme/ThemeClientContextProvider"
const Controls = () => {
const { theme, setTheme } = useTheme()
const switchThemes = () => {
if (theme === "dark") {
setTheme("light")
} else {
setTheme("dark")
}
}
return (
<div className={styles.wrapper}>
<Select scale={0.5} h="28px" pure onChange={switchThemes} value={theme}>
<Select.Option value="light">
<span className={styles.selectContent}>
<SunIcon size={14} /> Light
</span>
</Select.Option>
<Select.Option value="dark">
<span className={styles.selectContent}>
<MoonIcon size={14} /> Dark
</span>
</Select.Option>
</Select>
</div>
)
}
export default React.memo(Controls)

View file

@ -1,6 +1,6 @@
"use client" "use client"
import { Page, useBodyScroll, useMediaQuery } from "@geist-ui/core/dist" import { useBodyScroll, useMediaQuery } from "@geist-ui/core/dist"
import { useEffect, useState } from "react" import { useEffect, useState } from "react"
import styles from "./header.module.css" import styles from "./header.module.css"
@ -174,7 +174,7 @@ const Header = ({ signedIn = false, isAdmin = false }) => {
const buttons = pages.map(getButton) const buttons = pages.map(getButton)
return ( return (
<Page.Header> <header>
<div className={styles.tabs}> <div className={styles.tabs}>
<div className={styles.buttons}>{buttons}</div> <div className={styles.buttons}>{buttons}</div>
</div> </div>
@ -189,7 +189,7 @@ const Header = ({ signedIn = false, isAdmin = false }) => {
{buttons} {buttons}
</div> </div>
)} )}
</Page.Header> </header>
) )
} }

View file

@ -1,9 +1,4 @@
"use client"
import { Tabs, Textarea } from "@geist-ui/core/dist"
import Image from "next/image" import Image from "next/image"
import styles from "./home.module.css"
// TODO:components/new-post/ move these styles
import markdownStyles from "app/(posts)/components/preview/preview.module.css"
import Card from "./card" import Card from "./card"
import DocumentTabs from "app/(posts)/components/tabs" import DocumentTabs from "app/(posts)/components/tabs"
const Home = ({ const Home = ({

View file

@ -12,7 +12,8 @@
} }
.warning { .warning {
background: #f33; background: var(--warning);
color: var(--bg);
} }
.error { .error {
@ -21,7 +22,7 @@
[data-theme="light"] .warning, [data-theme="light"] .warning,
[data-theme="light"] .error { [data-theme="light"] .error {
color: var(--bg); color: var(--fg);
} }
.type { .type {

View file

@ -1,7 +1,7 @@
"use client" "use client"
import styles from "./post-list.module.css" import styles from "./post-list.module.css"
import ListItemSkeleton from "./list-item-skeleton" import { ListItemSkeleton } from "./list-item-skeleton"
import ListItem from "./list-item" import ListItem from "./list-item"
import { ChangeEvent, useCallback, useEffect, useState } from "react" import { ChangeEvent, useCallback, useEffect, useState } from "react"
import useDebounce from "@lib/hooks/use-debounce" import useDebounce from "@lib/hooks/use-debounce"
@ -9,6 +9,7 @@ import Link from "@components/link"
import type { PostWithFiles } from "@lib/server/prisma" import type { PostWithFiles } from "@lib/server/prisma"
import Input from "@components/input" import Input from "@components/input"
import Button from "@components/button" import Button from "@components/button"
import { useToasts } from "@components/toasts"
type Props = { type Props = {
initialPosts: string | PostWithFiles[] initialPosts: string | PostWithFiles[]
@ -29,6 +30,7 @@ const PostList = ({
const [posts, setPosts] = useState<PostWithFiles[]>(initialPosts) const [posts, setPosts] = useState<PostWithFiles[]>(initialPosts)
const [searching, setSearching] = useState(false) const [searching, setSearching] = useState(false)
const [hasMorePosts, setHasMorePosts] = useState(morePosts) const [hasMorePosts, setHasMorePosts] = useState(morePosts)
const { setToast } = useToasts()
const debouncedSearchValue = useDebounce(search, 200) const debouncedSearchValue = useDebounce(search, 200)
@ -71,7 +73,7 @@ const PostList = ({
} }
) )
const json = await res.json() const json = await res.json()
setPosts(json.posts) setPosts(json)
setSearching(false) setSearching(false)
} }
fetchPosts() fetchPosts()
@ -97,9 +99,13 @@ const PostList = ({
return return
} else { } else {
setPosts((posts) => posts.filter((post) => post.id !== postId)) setPosts((posts) => posts.filter((post) => post.id !== postId))
setToast({
message: "Post deleted",
type: "success"
})
} }
}, },
[] [setToast]
) )
return ( return (
@ -108,21 +114,18 @@ const PostList = ({
<Input <Input
placeholder="Search..." placeholder="Search..."
onChange={handleSearchChange} onChange={handleSearchChange}
disabled={Boolean(!posts?.length)} disabled={!posts}
style={{ maxWidth: 300 }} style={{ maxWidth: 300 }}
aria-label="Search"
/> />
</div> </div>
{!posts && <p style={{ color: "var(--warning)" }}>Failed to load.</p>} {!posts && <p style={{ color: "var(--warning)" }}>Failed to load.</p>}
{!posts?.length && searching && ( {/* {!posts?.length && (
<ul> <ul>
<li>
<ListItemSkeleton /> <ListItemSkeleton />
</li>
<li>
<ListItemSkeleton /> <ListItemSkeleton />
</li>
</ul> </ul>
)} )} */}
{posts?.length === 0 && posts && ( {posts?.length === 0 && posts && (
<p> <p>
No posts found. Create one{" "} No posts found. Create one{" "}

View file

@ -1,25 +1,22 @@
import styles from "./list-item.module.css"
import Card from "@components/card" import Card from "@components/card"
import Skeleton from "@components/skeleton" import Skeleton from "@components/skeleton"
import { Divider, Grid, Spacer } from "@geist-ui/core/dist"
const ListItemSkeleton = () => ( export const ListItemSkeleton = () => (
<Card> <li>
<Spacer height={1 / 2} /> <Card style={{ overflowY: "scroll" }}>
<Grid.Container justify={"space-between"} marginBottom={1 / 2}> <>
<Grid xs={8} paddingLeft={1 / 2}> <div className={styles.title}>
<Skeleton width={150} /> {/* title */}
</Grid> <Skeleton width={80} height={32} />
<Grid xs={7}> </div>
<Skeleton width={100} />
</Grid>
<Grid xs={4}>
<Skeleton width={70} />
</Grid>
</Grid.Container>
<Divider h="1px" my={0} /> <div className={styles.badges}>
<Skeleton width={200} /> <Skeleton width={30} height={32} />
</div>
</>
<hr />
<Skeleton width={100} height={32} />
</Card> </Card>
</li>
) )
export default ListItemSkeleton

View file

@ -0,0 +1,3 @@
import styles from './spinner.module.css'
export const Spinner = () => <div className={styles.spinner} />

View file

@ -0,0 +1,17 @@
.spinner {
border: 4px solid var(--light-gray);
border-top: 4px solid var(--gray);
border-radius: 50%;
width: 24px;
height: 24px;
animation: spin 1s linear infinite;
}
@keyframes spin {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}

View file

@ -0,0 +1,69 @@
"use client"
import Toast, { Toaster } from "react-hot-toast"
export type ToastType = "success" | "error" | "loading" | "default"
export type ToastProps = {
id?: string
type: ToastType
message: string
duration?: number
icon?: string
style?: React.CSSProperties
className?: string
loading?: boolean
loadingProgress?: number
}
export const useToasts = () => {
const setToast = (toast: ToastProps) => {
const { type, message, ...rest } = toast
if (toast.id) {
Toast.dismiss(toast.id)
}
switch (type) {
case "success":
Toast.success(message, rest)
break
case "error":
Toast.error(message, rest)
break
case "loading":
Toast.loading(message, rest)
break
default:
Toast(message, rest)
break
}
}
return { setToast }
}
export const Toasts = () => {
return (
<Toaster
position="bottom-right"
toastOptions={{
error: {
style: {
background: "var(--warning)",
color: "#fff"
}
},
success: {
style: {
background: "var(--light-gray)",
color: "var(--fg)"
}
},
iconTheme: {
primary: "var(--fg)",
secondary: "var(--bg)"
}
}}
/>
)
}

View file

@ -4,7 +4,6 @@ import styles from "@styles/Home.module.css"
import { getSession } from "@lib/server/session" import { getSession } from "@lib/server/session"
import ThemeProvider from "@components/theme/ThemeProvider" import ThemeProvider from "@components/theme/ThemeProvider"
import { THEME_COOKIE_NAME } from "@components/theme/theme" import { THEME_COOKIE_NAME } from "@components/theme/theme"
import { useServerTheme } from "@components/theme/ThemeServerContextProvider"
interface RootLayoutProps { interface RootLayoutProps {
children: React.ReactNode children: React.ReactNode

View file

@ -2,7 +2,9 @@
import Header from "@components/header" import Header from "@components/header"
import Page from "@components/page" import Page from "@components/page"
import { Toasts } from "@components/toasts"
import * as RadixTooltip from "@radix-ui/react-tooltip" import * as RadixTooltip from "@radix-ui/react-tooltip"
import { Toaster } from "react-hot-toast"
export function LayoutWrapper({ export function LayoutWrapper({
children, children,
@ -15,6 +17,7 @@ export function LayoutWrapper({
}) { }) {
return ( return (
<RadixTooltip.Provider delayDuration={200}> <RadixTooltip.Provider delayDuration={200}>
<Toasts />
<Page> <Page>
<Header isAdmin={isAdmin} signedIn={signedIn} /> <Header isAdmin={isAdmin} signedIn={signedIn} />
{children} {children}

View file

@ -1,134 +0,0 @@
"use client"
import { Input, Button, useToasts } from "@geist-ui/core/dist"
import { useState } from "react"
const Password = () => {
const [password, setPassword] = useState<string>("")
const [newPassword, setNewPassword] = useState<string>("")
const [confirmPassword, setConfirmPassword] = useState<string>("")
const { setToast } = useToasts()
const handlePasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setPassword(e.target.value)
}
const handleNewPasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setNewPassword(e.target.value)
}
const handleConfirmPasswordChange = (
e: React.ChangeEvent<HTMLInputElement>
) => {
setConfirmPassword(e.target.value)
}
const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault()
if (!password || !newPassword || !confirmPassword) {
setToast({
text: "Please fill out all fields",
type: "error"
})
}
if (newPassword !== confirmPassword) {
setToast({
text: "New password and confirm password do not match",
type: "error"
})
}
const res = await fetch("/server-api/auth/change-password", {
method: "PUT",
headers: {
"Content-Type": "application/json"
},
body: JSON.stringify({
oldPassword: password,
newPassword
})
})
if (res.status === 200) {
setToast({
text: "Password updated successfully",
type: "success"
})
setPassword("")
setNewPassword("")
setConfirmPassword("")
} else {
const data = await res.json()
setToast({
text: data.error ?? "Failed to update password",
type: "error"
})
}
}
return (
<form
style={{
display: "flex",
flexDirection: "column",
gap: "var(--gap)",
maxWidth: "300px"
}}
onSubmit={onSubmit}
>
<div>
<label htmlFor="current-password">Current password</label>
<Input
onChange={handlePasswordChange}
minLength={6}
maxLength={128}
value={password}
id="current-password"
htmlType="password"
required
autoComplete="current-password"
placeholder="Current Password"
width={"100%"}
/>
</div>
<div>
<label htmlFor="new-password">New password</label>
<Input
onChange={handleNewPasswordChange}
minLength={6}
maxLength={128}
value={newPassword}
id="new-password"
htmlType="password"
required
autoComplete="new-password"
placeholder="New Password"
width={"100%"}
/>
</div>
<div>
<label htmlFor="confirm-password">Confirm password</label>
<Input
onChange={handleConfirmPasswordChange}
minLength={6}
maxLength={128}
value={confirmPassword}
id="confirm-password"
htmlType="password"
required
autoComplete="confirm-password"
placeholder="Confirm Password"
width={"100%"}
/>
</div>
<Button htmlType="submit" auto>
Change Password
</Button>
</form>
)
}
export default Password

View file

@ -93,7 +93,7 @@ const Profile = ({ user }: { user: User }) => {
disabled disabled
/> />
</div> </div>
<div> {/* <div>
<label htmlFor="bio">Biography (max 250 characters)</label> <label htmlFor="bio">Biography (max 250 characters)</label>
<textarea <textarea
id="bio" id="bio"
@ -103,8 +103,8 @@ const Profile = ({ user }: { user: User }) => {
value={bio} value={bio}
onChange={handleBioChange} onChange={handleBioChange}
/> />
</div> </div> */}
<Button htmlType="submit" auto> <Button type="submit">
Submit Submit
</Button> </Button>
</form> </form>

View file

@ -45,7 +45,7 @@
--gray-alpha: rgba(255, 255, 255, 0.5); --gray-alpha: rgba(255, 255, 255, 0.5);
--selection: rgba(255, 255, 255, 0.99); --selection: rgba(255, 255, 255, 0.99);
--border: var(--lighter-gray); --border: var(--lighter-gray);
--warning: rgb(27, 134, 23); --warning: #ff6700;
--link: #3291ff; --link: #3291ff;
color-scheme: dark; color-scheme: dark;
} }
@ -78,8 +78,9 @@
box-sizing: border-box; box-sizing: border-box;
} }
// TODO: replace this with an accessible alternative
*:focus-visible { *:focus-visible {
outline: 1px solid var(--gray); outline: none;
} }
::selection { ::selection {
@ -179,3 +180,8 @@ textarea {
padding: var(--gap-half); padding: var(--gap-half);
border-radius: var(--radius); border-radius: var(--radius);
} }
textarea:focus {
border-color: var(--light-gray);
outline: none;
}

View file

@ -18,6 +18,7 @@
"@next-auth/prisma-adapter": "^1.0.5", "@next-auth/prisma-adapter": "^1.0.5",
"@next/eslint-plugin-next": "13.0.5-canary.3", "@next/eslint-plugin-next": "13.0.5-canary.3",
"@prisma/client": "^4.6.1", "@prisma/client": "^4.6.1",
"@radix-ui/react-dropdown-menu": "^2.0.1",
"@radix-ui/react-popover": "^1.0.2", "@radix-ui/react-popover": "^1.0.2",
"@radix-ui/react-tabs": "^1.0.1", "@radix-ui/react-tabs": "^1.0.1",
"@radix-ui/react-tooltip": "^1.0.2", "@radix-ui/react-tooltip": "^1.0.2",
@ -26,8 +27,8 @@
"client-zip": "2.2.1", "client-zip": "2.2.1",
"cookies-next": "^2.1.1", "cookies-next": "^2.1.1",
"jest": "^29.3.1", "jest": "^29.3.1",
"next": "13.0.5-canary.3", "next": "13.0.6-canary.1",
"next-auth": "^4.16.4", "next-auth": "^4.17.0",
"rc-table": "7.24.1", "rc-table": "7.24.1",
"react": "18.2.0", "react": "18.2.0",
"react-datepicker": "4.8.0", "react-datepicker": "4.8.0",

View file

@ -7,6 +7,7 @@ specifiers:
'@next/bundle-analyzer': 12.1.6 '@next/bundle-analyzer': 12.1.6
'@next/eslint-plugin-next': 13.0.5-canary.3 '@next/eslint-plugin-next': 13.0.5-canary.3
'@prisma/client': ^4.6.1 '@prisma/client': ^4.6.1
'@radix-ui/react-dropdown-menu': ^2.0.1
'@radix-ui/react-popover': ^1.0.2 '@radix-ui/react-popover': ^1.0.2
'@radix-ui/react-tabs': ^1.0.1 '@radix-ui/react-tabs': ^1.0.1
'@radix-ui/react-tooltip': ^1.0.2 '@radix-ui/react-tooltip': ^1.0.2
@ -25,8 +26,8 @@ specifiers:
eslint-config-next: 13.0.3 eslint-config-next: 13.0.3
jest: ^29.3.1 jest: ^29.3.1
katex: ^0.16.3 katex: ^0.16.3
next: 13.0.5-canary.3 next: 13.0.6-canary.1
next-auth: ^4.16.4 next-auth: ^4.17.0
next-unused: 0.0.6 next-unused: 0.0.6
prettier: 2.6.2 prettier: 2.6.2
prisma: ^4.6.1 prisma: ^4.6.1
@ -47,9 +48,10 @@ specifiers:
dependencies: dependencies:
'@geist-ui/core': 2.3.8_biqbaboplfbrettd7655fr4n2y '@geist-ui/core': 2.3.8_biqbaboplfbrettd7655fr4n2y
'@geist-ui/icons': 1.0.2_zhza2kbnl2wkkf7vqdl3ton2f4 '@geist-ui/icons': 1.0.2_zhza2kbnl2wkkf7vqdl3ton2f4
'@next-auth/prisma-adapter': 1.0.5_2pl3b2nwmjya7el2zbe6cwkney '@next-auth/prisma-adapter': 1.0.5_o53gfpk3vz2btjrokqfjjwwn3m
'@next/eslint-plugin-next': 13.0.5-canary.3 '@next/eslint-plugin-next': 13.0.5-canary.3
'@prisma/client': 4.6.1_prisma@4.6.1 '@prisma/client': 4.6.1_prisma@4.6.1
'@radix-ui/react-dropdown-menu': 2.0.1_jbvntnid6ohjelon6ccj5dhg2u
'@radix-ui/react-popover': 1.0.2_jbvntnid6ohjelon6ccj5dhg2u '@radix-ui/react-popover': 1.0.2_jbvntnid6ohjelon6ccj5dhg2u
'@radix-ui/react-tabs': 1.0.1_biqbaboplfbrettd7655fr4n2y '@radix-ui/react-tabs': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-tooltip': 1.0.2_jbvntnid6ohjelon6ccj5dhg2u '@radix-ui/react-tooltip': 1.0.2_jbvntnid6ohjelon6ccj5dhg2u
@ -58,8 +60,8 @@ dependencies:
client-zip: 2.2.1 client-zip: 2.2.1
cookies-next: 2.1.1 cookies-next: 2.1.1
jest: 29.3.1_@types+node@17.0.23 jest: 29.3.1_@types+node@17.0.23
next: 13.0.5-canary.3_biqbaboplfbrettd7655fr4n2y next: 13.0.6-canary.1_biqbaboplfbrettd7655fr4n2y
next-auth: 4.16.4_p6ldyhl3eilwbxhfzedqikw7ni next-auth: 4.17.0_cejjzyjft5qpe7pbv5t5jzassa
rc-table: 7.24.1_biqbaboplfbrettd7655fr4n2y rc-table: 7.24.1_biqbaboplfbrettd7655fr4n2y
react: 18.2.0 react: 18.2.0
react-datepicker: 4.8.0_biqbaboplfbrettd7655fr4n2y react-datepicker: 4.8.0_biqbaboplfbrettd7655fr4n2y
@ -810,14 +812,14 @@ packages:
- supports-color - supports-color
dev: false dev: false
/@next-auth/prisma-adapter/1.0.5_2pl3b2nwmjya7el2zbe6cwkney: /@next-auth/prisma-adapter/1.0.5_o53gfpk3vz2btjrokqfjjwwn3m:
resolution: {integrity: sha512-VqMS11IxPXrPGXw6Oul6jcyS/n8GLOWzRMrPr3EMdtD6eOalM6zz05j08PcNiis8QzkfuYnCv49OvufTuaEwYQ==} resolution: {integrity: sha512-VqMS11IxPXrPGXw6Oul6jcyS/n8GLOWzRMrPr3EMdtD6eOalM6zz05j08PcNiis8QzkfuYnCv49OvufTuaEwYQ==}
peerDependencies: peerDependencies:
'@prisma/client': '>=2.26.0 || >=3' '@prisma/client': '>=2.26.0 || >=3'
next-auth: ^4 next-auth: ^4
dependencies: dependencies:
'@prisma/client': 4.6.1_prisma@4.6.1 '@prisma/client': 4.6.1_prisma@4.6.1
next-auth: 4.16.4_p6ldyhl3eilwbxhfzedqikw7ni next-auth: 4.17.0_cejjzyjft5qpe7pbv5t5jzassa
dev: false dev: false
/@next/bundle-analyzer/12.1.6: /@next/bundle-analyzer/12.1.6:
@ -829,8 +831,8 @@ packages:
- utf-8-validate - utf-8-validate
dev: true dev: true
/@next/env/13.0.5-canary.3: /@next/env/13.0.6-canary.1:
resolution: {integrity: sha512-x8hWJis9ICO6y5e7BNBPjFmFvgDRe5URd5YV3qgkw1flCRDLL3QtfTLig6nXH9DaBHaQ1a0ej7vvlzyDq6iHEg==} resolution: {integrity: sha512-L88vP2GvU2NvF5YSIDjqYzzJQRNDg3F08qE/pTLClsYXLusRWAJ1lgI9sZFqLrMbZuj2xd0hWYyYjCrrg/LLLw==}
dev: false dev: false
/@next/eslint-plugin-next/13.0.3: /@next/eslint-plugin-next/13.0.3:
@ -845,8 +847,8 @@ packages:
glob: 7.1.7 glob: 7.1.7
dev: false dev: false
/@next/swc-android-arm-eabi/13.0.5-canary.3: /@next/swc-android-arm-eabi/13.0.6-canary.1:
resolution: {integrity: sha512-t/HW4YMqsXMnyaOvgK4U5wue8sgV6+yjp7eNKVqLCDxMdXemLUIqPTdVOdfU2oVScUNYj5VQFWO52HE/x6LugQ==} resolution: {integrity: sha512-3/ZSIsm2yYZXUyqupAri4+6J97wVdHIP2yrcTLoKdYQKm1lp1z5hIZLtxQCV4vLyYNlNIafvIqeasfYWuYzCgw==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm] cpu: [arm]
os: [android] os: [android]
@ -854,8 +856,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-android-arm64/13.0.5-canary.3: /@next/swc-android-arm64/13.0.6-canary.1:
resolution: {integrity: sha512-WLNjvoC9sDWqDgYOJDEPKeYwS/tymTvUDBDzQfGyl0XOW8g01wD4S1TBlHDP+D9y85BxLdwlUJoYbOwhAfwI0A==} resolution: {integrity: sha512-/J2EXT8L82HgV8sf2SzSdpTuT5gtpPGgekPRCF1ttYyN8FAR/VHPKOLYTbKz3NQ95ogttrW28RFPsMSekQpLaQ==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm64] cpu: [arm64]
os: [android] os: [android]
@ -863,8 +865,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-darwin-arm64/13.0.5-canary.3: /@next/swc-darwin-arm64/13.0.6-canary.1:
resolution: {integrity: sha512-JeGAnktTy7fQeF8PrQJgT25LlNnRFiH6zXWVfHTRk4ufcERifaOnwpzPLjIxF6euo3qD8VynCQhYH/JsVS5HNQ==} resolution: {integrity: sha512-n+IxXIJSEm3HdumauULSZCehyKyQ0EfWU/qf5oMjyGOQ/sHQfgT5sGYtfnnNvvhSlVgFhyz7MalPRwJQWAIIZQ==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm64] cpu: [arm64]
os: [darwin] os: [darwin]
@ -872,8 +874,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-darwin-x64/13.0.5-canary.3: /@next/swc-darwin-x64/13.0.6-canary.1:
resolution: {integrity: sha512-yehsw+60uE47but0FUJD2+/06tQSsIm1/XFlFFhHv4d+bq39GB2fT+wL+h/aLK+es0yY7rx1+6Yad7CQ7a2J+Q==} resolution: {integrity: sha512-0gQ2zLBFJT8rO080APvY/fHxjSIRVD7VpF0C12BXroFJ3XMnCoa6scniVgKwV4pjhTtlJoLSk3y4wNITyouLDQ==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [x64] cpu: [x64]
os: [darwin] os: [darwin]
@ -881,8 +883,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-freebsd-x64/13.0.5-canary.3: /@next/swc-freebsd-x64/13.0.6-canary.1:
resolution: {integrity: sha512-Nd1RicUhogxTsassJNGG+4ha5ezyjJb58J6Vjs9kT5I40Iu8Mw98hCwn6VmiCtJMpY4HGJcuYtCYyXh5hY6ByQ==} resolution: {integrity: sha512-zAnm1JtxgrNkjiBef7U8lO9JOqNhy13mKA0Z27XCeAunhaar0hNZNHnKUlISVhAZovpHs96fp0mm7Skw0WCkpA==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [x64] cpu: [x64]
os: [freebsd] os: [freebsd]
@ -890,8 +892,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-linux-arm-gnueabihf/13.0.5-canary.3: /@next/swc-linux-arm-gnueabihf/13.0.6-canary.1:
resolution: {integrity: sha512-t8TNq1rVBaYJz5dzAJIXgfYh7KRFich6gIT38zJ82McbKWFccuY6+9+FP6n0zkNFAZAd0CBvTQJ8fQFX6XPMog==} resolution: {integrity: sha512-JJCdGBEeNzGXm0AQUfVY9x3C3WvxCcawZ1tCYYkOqLD21Dj5gDTNSZT2cAGFDovfyn5hN/Vltf20Zs8ilUPr1w==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm] cpu: [arm]
os: [linux] os: [linux]
@ -899,8 +901,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-linux-arm64-gnu/13.0.5-canary.3: /@next/swc-linux-arm64-gnu/13.0.6-canary.1:
resolution: {integrity: sha512-jsHOBJSkBn247KlW2BG5j1hoLG3EIa1gE/FLCj7w/cQBCtrhb83oGBxat52Wh7UmwajrvNl13clQV0JlmkX3sg==} resolution: {integrity: sha512-oE9D+Mm240fRpZWZBhFiztmJ/D7uwQ3jfOZcQVf1aN3P+9De+5jyv2TKoHrjrRh5xjloTEDIpt2wNKimlwZvkg==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm64] cpu: [arm64]
os: [linux] os: [linux]
@ -908,8 +910,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-linux-arm64-musl/13.0.5-canary.3: /@next/swc-linux-arm64-musl/13.0.6-canary.1:
resolution: {integrity: sha512-AVsfjFzRKx9I4rbHBOK1WZyl3jF635uLcb7ssrJtxyMGHcr4R9KCTcl+kv7RZ2WNKVafun0pUAITKHoqKVO4fg==} resolution: {integrity: sha512-+DukK2LdoZo4Ds0o2XGAC9324O9auxvHMF4MwDzrXrherBFEeSZUDIxcz4zu5/pd8dyQ8JY9nwILJEevla23pA==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm64] cpu: [arm64]
os: [linux] os: [linux]
@ -917,8 +919,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-linux-x64-gnu/13.0.5-canary.3: /@next/swc-linux-x64-gnu/13.0.6-canary.1:
resolution: {integrity: sha512-E83c9cs+8hwvpTUcn991ec/Q6tDG54SzvVrvNqAj23yiWGfHx0MDt1NqYtF0WOJ7W+6G6PtlD9SR0KKiaPMSeA==} resolution: {integrity: sha512-1bULat6aif5E48s23lAh4vmuVogxz3arXMawlqHwHqwJWpzxF2vyAl3Ilh4mXJnD1xvwiGRH7Aov7HMQJhF2+g==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [x64] cpu: [x64]
os: [linux] os: [linux]
@ -926,8 +928,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-linux-x64-musl/13.0.5-canary.3: /@next/swc-linux-x64-musl/13.0.6-canary.1:
resolution: {integrity: sha512-d9CT6qNDxf/C2j8oU6sb2XK2C0qE5N6uI+JL2r19fZKgQIgjCMEWbuyI2oVpq/LrwJ4w0vmjxuRg/9YEb8zX9w==} resolution: {integrity: sha512-LN9qXV3Rh1jMcaSXg5DGZYbN6nVCn6A/aQBTw4K5DWu7bHTiEkq4d7/OZN+gfJcqW8BuRtG4zLeraae/1RO9pw==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [x64] cpu: [x64]
os: [linux] os: [linux]
@ -935,8 +937,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-win32-arm64-msvc/13.0.5-canary.3: /@next/swc-win32-arm64-msvc/13.0.6-canary.1:
resolution: {integrity: sha512-4Mcw9pc8HgZ0FoOybyfDW17dXXgJnqs8TDXbWDlV9tGrbYJ5yPPe2r0gZaPCGCtbYLLz3XQfd6kDDf2J0hx7Kw==} resolution: {integrity: sha512-792QCYO+lF4N1c2bwYVwhLwMJMhnmvQRVOnCmRlGhbaZJ0f6ZuIYCxjqAqV8Por+u0ErenRMv5BGleK8wHfQJA==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [arm64] cpu: [arm64]
os: [win32] os: [win32]
@ -944,8 +946,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-win32-ia32-msvc/13.0.5-canary.3: /@next/swc-win32-ia32-msvc/13.0.6-canary.1:
resolution: {integrity: sha512-vGHnW+l+aE264lmz7warC7FPEUb3RDypooKZHDb7fTtlbqnEBf8MVlPYLVj73MSwbW8vKfoatScCG09tljYztA==} resolution: {integrity: sha512-hwL9JjT7C1ZP+Wj7F4QIoewEErqc70Ax0aeKqQZdKU57djmIWvvvnyh9p/uUQFHLc3M4Um+XvtqpyFZB6RcYbg==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [ia32] cpu: [ia32]
os: [win32] os: [win32]
@ -953,8 +955,8 @@ packages:
dev: false dev: false
optional: true optional: true
/@next/swc-win32-x64-msvc/13.0.5-canary.3: /@next/swc-win32-x64-msvc/13.0.6-canary.1:
resolution: {integrity: sha512-BvqxBWnY2D0DDBh+NxSQ70+f/kLEEQAbkbyjOcKIxjtYY9Gu6FHAHHyyTNwXMRKr7X3yiX/olLTy1EekCZ0M6A==} resolution: {integrity: sha512-KA6/9W7ZEfkN/ecH8f/fg+WyQaJnRG/UE2IKOdQHbL4uVtHJk5PQmYT+nBZFVnbNHJB2ouNmL4pFJM/7JjNBww==}
engines: {node: '>= 10'} engines: {node: '>= 10'}
cpu: [x64] cpu: [x64]
os: [win32] os: [win32]
@ -1092,6 +1094,26 @@ packages:
react-dom: 18.2.0_react@18.2.0 react-dom: 18.2.0_react@18.2.0
dev: false dev: false
/@radix-ui/react-dropdown-menu/2.0.1_jbvntnid6ohjelon6ccj5dhg2u:
resolution: {integrity: sha512-WDZqmwsNuxdBMkvgB85UeSPAT0wSBd+ojxtzX7lU7uYYh47gacCj6Spo0l9+X4TMe3JA1BBMN9c7OhIMaQeKbg==}
peerDependencies:
react: ^16.8 || ^17.0 || ^18.0
react-dom: ^16.8 || ^17.0 || ^18.0
dependencies:
'@babel/runtime': 7.20.1
'@radix-ui/primitive': 1.0.0
'@radix-ui/react-compose-refs': 1.0.0_react@18.2.0
'@radix-ui/react-context': 1.0.0_react@18.2.0
'@radix-ui/react-id': 1.0.0_react@18.2.0
'@radix-ui/react-menu': 2.0.1_jbvntnid6ohjelon6ccj5dhg2u
'@radix-ui/react-primitive': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-use-controllable-state': 1.0.0_react@18.2.0
react: 18.2.0
react-dom: 18.2.0_react@18.2.0
transitivePeerDependencies:
- '@types/react'
dev: false
/@radix-ui/react-focus-guards/1.0.0_react@18.2.0: /@radix-ui/react-focus-guards/1.0.0_react@18.2.0:
resolution: {integrity: sha512-UagjDk4ijOAnGu4WMUPj9ahi7/zJJqNZ9ZAiGPp7waUWJO0O1aWXi/udPphI0IUjvrhBsZJGSN66dR2dsueLWQ==} resolution: {integrity: sha512-UagjDk4ijOAnGu4WMUPj9ahi7/zJJqNZ9ZAiGPp7waUWJO0O1aWXi/udPphI0IUjvrhBsZJGSN66dR2dsueLWQ==}
peerDependencies: peerDependencies:
@ -1125,6 +1147,37 @@ packages:
react: 18.2.0 react: 18.2.0
dev: false dev: false
/@radix-ui/react-menu/2.0.1_jbvntnid6ohjelon6ccj5dhg2u:
resolution: {integrity: sha512-I5FFZQxCl2fHoJ7R0m5/oWA9EX8/ttH4AbgneoCH7DAXQioFeb0XMAYnOVSp1GgJZ1Nx/mohxNQSeTMcaF1YPw==}
peerDependencies:
react: ^16.8 || ^17.0 || ^18.0
react-dom: ^16.8 || ^17.0 || ^18.0
dependencies:
'@babel/runtime': 7.20.1
'@radix-ui/primitive': 1.0.0
'@radix-ui/react-collection': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-compose-refs': 1.0.0_react@18.2.0
'@radix-ui/react-context': 1.0.0_react@18.2.0
'@radix-ui/react-direction': 1.0.0_react@18.2.0
'@radix-ui/react-dismissable-layer': 1.0.2_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-focus-guards': 1.0.0_react@18.2.0
'@radix-ui/react-focus-scope': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-id': 1.0.0_react@18.2.0
'@radix-ui/react-popper': 1.0.1_jbvntnid6ohjelon6ccj5dhg2u
'@radix-ui/react-portal': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-presence': 1.0.0_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-primitive': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-roving-focus': 1.0.1_biqbaboplfbrettd7655fr4n2y
'@radix-ui/react-slot': 1.0.1_react@18.2.0
'@radix-ui/react-use-callback-ref': 1.0.0_react@18.2.0
aria-hidden: 1.2.1_ulhmhxukhxjgxaybrsjlob7ffu
react: 18.2.0
react-dom: 18.2.0_react@18.2.0
react-remove-scroll: 2.5.5_ulhmhxukhxjgxaybrsjlob7ffu
transitivePeerDependencies:
- '@types/react'
dev: false
/@radix-ui/react-popover/1.0.2_jbvntnid6ohjelon6ccj5dhg2u: /@radix-ui/react-popover/1.0.2_jbvntnid6ohjelon6ccj5dhg2u:
resolution: {integrity: sha512-4tqZEl9w95R5mlZ/sFdgBnfhCBOEPepLIurBA5kt/qaAhldJ1tNQd0ngr0ET0AHbPotT4mwxMPr7a+MA/wbK0g==} resolution: {integrity: sha512-4tqZEl9w95R5mlZ/sFdgBnfhCBOEPepLIurBA5kt/qaAhldJ1tNQd0ngr0ET0AHbPotT4mwxMPr7a+MA/wbK0g==}
peerDependencies: peerDependencies:
@ -5161,8 +5214,8 @@ packages:
dev: true dev: true
optional: true optional: true
/next-auth/4.16.4_p6ldyhl3eilwbxhfzedqikw7ni: /next-auth/4.17.0_cejjzyjft5qpe7pbv5t5jzassa:
resolution: {integrity: sha512-KXW578+ER1u5RcWLwCHMdb/RIBIO6JM8r6xlf9RIPSKzkvDcX9FHiZfJS2vOq/SurHXPJZc4J3OS4IDJpF74Dw==} resolution: {integrity: sha512-aN2tdnjS0MDeUpB2tBDOaWnegkgeMWrsccujbXRGMJ607b+EwRcy63MFGSr0OAboDJEe0902piXQkt94GqF8Qw==}
engines: {node: ^12.19.0 || ^14.15.0 || ^16.13.0 || ^18.12.0} engines: {node: ^12.19.0 || ^14.15.0 || ^16.13.0 || ^18.12.0}
peerDependencies: peerDependencies:
next: ^12.2.5 || ^13 next: ^12.2.5 || ^13
@ -5177,7 +5230,7 @@ packages:
'@panva/hkdf': 1.0.2 '@panva/hkdf': 1.0.2
cookie: 0.5.0 cookie: 0.5.0
jose: 4.11.0 jose: 4.11.0
next: 13.0.5-canary.3_biqbaboplfbrettd7655fr4n2y next: 13.0.6-canary.1_biqbaboplfbrettd7655fr4n2y
oauth: 0.9.15 oauth: 0.9.15
openid-client: 5.3.0 openid-client: 5.3.0
preact: 10.11.2 preact: 10.11.2
@ -5198,8 +5251,8 @@ packages:
- supports-color - supports-color
dev: true dev: true
/next/13.0.5-canary.3_biqbaboplfbrettd7655fr4n2y: /next/13.0.6-canary.1_biqbaboplfbrettd7655fr4n2y:
resolution: {integrity: sha512-u3As6SkLXf2u9Mt06B3gBdQzVi45qoH27fyj2UGH+GcWjloz0x0Q+99CEz0sr93zS6iQRZuDn93PMZXoXIa8Og==} resolution: {integrity: sha512-aLHWxU8tMcsVKyStTQvfATZ8TzX/X/VbfV1erVn+pF4uI5+ETvAGWO1geo1FrLzrW+xB/h+9UPAUiqh21oVZEA==}
engines: {node: '>=14.6.0'} engines: {node: '>=14.6.0'}
hasBin: true hasBin: true
peerDependencies: peerDependencies:
@ -5216,28 +5269,27 @@ packages:
sass: sass:
optional: true optional: true
dependencies: dependencies:
'@next/env': 13.0.5-canary.3 '@next/env': 13.0.6-canary.1
'@swc/helpers': 0.4.14 '@swc/helpers': 0.4.14
caniuse-lite: 1.0.30001431 caniuse-lite: 1.0.30001431
postcss: 8.4.14 postcss: 8.4.14
react: 18.2.0 react: 18.2.0
react-dom: 18.2.0_react@18.2.0 react-dom: 18.2.0_react@18.2.0
styled-jsx: 5.1.0_react@18.2.0 styled-jsx: 5.1.0_react@18.2.0
use-sync-external-store: 1.2.0_react@18.2.0
optionalDependencies: optionalDependencies:
'@next/swc-android-arm-eabi': 13.0.5-canary.3 '@next/swc-android-arm-eabi': 13.0.6-canary.1
'@next/swc-android-arm64': 13.0.5-canary.3 '@next/swc-android-arm64': 13.0.6-canary.1
'@next/swc-darwin-arm64': 13.0.5-canary.3 '@next/swc-darwin-arm64': 13.0.6-canary.1
'@next/swc-darwin-x64': 13.0.5-canary.3 '@next/swc-darwin-x64': 13.0.6-canary.1
'@next/swc-freebsd-x64': 13.0.5-canary.3 '@next/swc-freebsd-x64': 13.0.6-canary.1
'@next/swc-linux-arm-gnueabihf': 13.0.5-canary.3 '@next/swc-linux-arm-gnueabihf': 13.0.6-canary.1
'@next/swc-linux-arm64-gnu': 13.0.5-canary.3 '@next/swc-linux-arm64-gnu': 13.0.6-canary.1
'@next/swc-linux-arm64-musl': 13.0.5-canary.3 '@next/swc-linux-arm64-musl': 13.0.6-canary.1
'@next/swc-linux-x64-gnu': 13.0.5-canary.3 '@next/swc-linux-x64-gnu': 13.0.6-canary.1
'@next/swc-linux-x64-musl': 13.0.5-canary.3 '@next/swc-linux-x64-musl': 13.0.6-canary.1
'@next/swc-win32-arm64-msvc': 13.0.5-canary.3 '@next/swc-win32-arm64-msvc': 13.0.6-canary.1
'@next/swc-win32-ia32-msvc': 13.0.5-canary.3 '@next/swc-win32-ia32-msvc': 13.0.6-canary.1
'@next/swc-win32-x64-msvc': 13.0.5-canary.3 '@next/swc-win32-x64-msvc': 13.0.6-canary.1
transitivePeerDependencies: transitivePeerDependencies:
- '@babel/core' - '@babel/core'
- babel-plugin-macros - babel-plugin-macros
@ -7100,14 +7152,6 @@ packages:
tslib: 2.4.1 tslib: 2.4.1
dev: false dev: false
/use-sync-external-store/1.2.0_react@18.2.0:
resolution: {integrity: sha512-eEgnFxGQ1Ife9bzYs6VLi8/4X6CObHMw9Qr9tPY43iKwsPw8xE8+EFsf/2cFZ5S3esXgpWgtSCtLNS41F+sKPA==}
peerDependencies:
react: ^16.8.0 || ^17.0.0 || ^18.0.0
dependencies:
react: 18.2.0
dev: false
/util-deprecate/1.0.2: /util-deprecate/1.0.2:
resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==}