feat: Converting space app to pages dir (#2052)
Co-authored-by: Bavisetti Narayan <72156168+NarayanBavisetti@users.noreply.github.com> Co-authored-by: NarayanBavisetti <narayan3119@gmail.com> Co-authored-by: Bavisetti Narayan <narayan@Bavisettis-MacBook-Pro.local> Co-authored-by: Nikhil <118773738+pablohashescobar@users.noreply.github.com> Co-authored-by: M. Palanikannan <73993394+Palanikannan1437@users.noreply.github.com> Co-authored-by: Lakhan Baheti <94619783+1akhanBaheti@users.noreply.github.com> Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com> Co-authored-by: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com>
This commit is contained in:
parent
c03550656a
commit
8a95a41100
133 changed files with 7993 additions and 1521 deletions
149
apps/space/components/ui/dropdown.tsx
Normal file
149
apps/space/components/ui/dropdown.tsx
Normal file
|
|
@ -0,0 +1,149 @@
|
|||
"use client";
|
||||
|
||||
import { Fragment, useState, useRef } from "react";
|
||||
|
||||
// next
|
||||
import Link from "next/link";
|
||||
|
||||
// headless
|
||||
import { Popover, Transition } from "@headlessui/react";
|
||||
import { ChevronLeftIcon, CheckIcon } from "@heroicons/react/20/solid";
|
||||
|
||||
// hooks
|
||||
import useOutSideClick from "hooks/use-outside-click";
|
||||
|
||||
type ItemOptionType = {
|
||||
display: React.ReactNode;
|
||||
as?: "button" | "link" | "div";
|
||||
href?: string;
|
||||
isSelected?: boolean;
|
||||
onClick?: () => void;
|
||||
children?: ItemOptionType[] | null;
|
||||
};
|
||||
|
||||
type DropdownItemProps = {
|
||||
item: ItemOptionType;
|
||||
};
|
||||
|
||||
type DropDownListProps = {
|
||||
open: boolean;
|
||||
handleClose?: () => void;
|
||||
items: ItemOptionType[];
|
||||
};
|
||||
|
||||
type DropdownProps = {
|
||||
button: React.ReactNode | (() => React.ReactNode);
|
||||
items: ItemOptionType[];
|
||||
};
|
||||
|
||||
const DropdownList: React.FC<DropDownListProps> = (props) => {
|
||||
const { open, items, handleClose } = props;
|
||||
|
||||
const ref = useRef(null);
|
||||
|
||||
useOutSideClick(ref, () => {
|
||||
if (handleClose) handleClose();
|
||||
});
|
||||
|
||||
return (
|
||||
<Popover className="absolute -left-1">
|
||||
<Transition
|
||||
show={open}
|
||||
as={Fragment}
|
||||
enter="transition ease-out duration-200"
|
||||
enterFrom="opacity-0 translate-y-1"
|
||||
enterTo="opacity-100 translate-y-0"
|
||||
leave="transition ease-in duration-150"
|
||||
leaveFrom="opacity-100 translate-y-0"
|
||||
leaveTo="opacity-0 translate-y-1"
|
||||
>
|
||||
<Popover.Panel
|
||||
ref={ref}
|
||||
className="absolute left-1/2 -translate-x-full z-10 mt-1 max-w-[9rem] origin-top-right select-none rounded-md bg-custom-background-90 border border-custom-border-300 text-xs shadow-lg focus:outline-none"
|
||||
>
|
||||
<div className="w-full text-sm rounded-md shadow-lg">
|
||||
{items.map((item, index) => (
|
||||
<DropdownItem key={index} item={item} />
|
||||
))}
|
||||
</div>
|
||||
</Popover.Panel>
|
||||
</Transition>
|
||||
</Popover>
|
||||
);
|
||||
};
|
||||
|
||||
const DropdownItem: React.FC<DropdownItemProps> = (props) => {
|
||||
const { item } = props;
|
||||
const { display, children, as: as_, href, onClick, isSelected } = item;
|
||||
|
||||
const [open, setOpen] = useState(false);
|
||||
|
||||
return (
|
||||
<div className="w-full group relative flex gap-x-6 rounded-lg p-1">
|
||||
{(!as_ || as_ === "button" || as_ === "div") && (
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => {
|
||||
if (!children) {
|
||||
if (onClick) onClick();
|
||||
return;
|
||||
}
|
||||
setOpen((prev) => !prev);
|
||||
}}
|
||||
className={`w-full flex items-center gap-1 rounded px-1 py-1.5 text-custom-text-200 hover:bg-custom-background-80 ${
|
||||
isSelected ? "bg-custom-background-80" : ""
|
||||
}`}
|
||||
>
|
||||
{children && <ChevronLeftIcon className="h-5 w-5 transition-transform transform" />}
|
||||
{!children && <span />}
|
||||
<span className="truncate text-xs">{display}</span>
|
||||
<CheckIcon className={`h-3.5 w-3.5 opacity-0 ${isSelected ? "opacity-100" : ""}`} />
|
||||
</button>
|
||||
)}
|
||||
|
||||
{as_ === "link" && <Link href={href || "#"}>{display}</Link>}
|
||||
|
||||
{children && <DropdownList open={open} handleClose={() => setOpen(false)} items={children} />}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
const Dropdown: React.FC<DropdownProps> = (props) => {
|
||||
const { button, items } = props;
|
||||
|
||||
return (
|
||||
<Popover className="relative">
|
||||
{({ open }) => (
|
||||
<>
|
||||
<Popover.Button
|
||||
className={`group flex items-center justify-between gap-2 rounded-md border border-custom-border-200 px-3 py-1.5 text-xs shadow-sm duration-300 focus:outline-none hover:text-custom-text-100 hover:bg-custom-background-90 ${
|
||||
open ? "bg-custom-background-90 text-custom-text-100" : "text-custom-text-200"
|
||||
}`}
|
||||
>
|
||||
{typeof button === "function" ? button() : button}
|
||||
</Popover.Button>
|
||||
|
||||
<Transition
|
||||
as={Fragment}
|
||||
enter="transition ease-out duration-200"
|
||||
enterFrom="opacity-0 translate-y-1"
|
||||
enterTo="opacity-100 translate-y-0"
|
||||
leave="transition ease-in duration-150"
|
||||
leaveFrom="opacity-100 translate-y-0"
|
||||
leaveTo="opacity-0 translate-y-1"
|
||||
>
|
||||
<Popover.Panel className="absolute left-full -translate-x-full z-10 mt-1 w-36 origin-top-right select-none rounded-md bg-custom-background-90 border border-custom-border-300 text-xs shadow-lg focus:outline-none">
|
||||
<div className="w-full">
|
||||
{items.map((item, index) => (
|
||||
<DropdownItem key={index} item={item} />
|
||||
))}
|
||||
</div>
|
||||
</Popover.Panel>
|
||||
</Transition>
|
||||
</>
|
||||
)}
|
||||
</Popover>
|
||||
);
|
||||
};
|
||||
|
||||
export { Dropdown };
|
||||
Loading…
Add table
Add a link
Reference in a new issue