fix: minor ui fixes (#488)

This commit is contained in:
Aaryan Khandelwal 2023-03-22 16:58:32 +05:30 committed by GitHub
parent 283950c8e2
commit 2e346158ba
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 305 additions and 202 deletions

View File

@ -21,7 +21,13 @@ import { CreateUpdateViewModal } from "components/views";
// ui // ui
import { Avatar, EmptySpace, EmptySpaceItem, PrimaryButton } from "components/ui"; import { Avatar, EmptySpace, EmptySpaceItem, PrimaryButton } from "components/ui";
// icons // icons
import { PlusIcon, RectangleStackIcon, TrashIcon, XMarkIcon } from "@heroicons/react/24/outline"; import {
ListBulletIcon,
PlusIcon,
RectangleStackIcon,
TrashIcon,
XMarkIcon,
} from "@heroicons/react/24/outline";
// helpers // helpers
import { getStatesList } from "helpers/state.helper"; import { getStatesList } from "helpers/state.helper";
// types // types
@ -613,6 +619,14 @@ export const IssuesView: React.FC<Props> = ({ type = "issue", openIssuesListModa
document.dispatchEvent(e); document.dispatchEvent(e);
}} }}
/> />
{openIssuesListModal && (
<EmptySpaceItem
title="Add an existing issue"
description="Open list"
Icon={ListBulletIcon}
action={openIssuesListModal}
/>
)}
</EmptySpace> </EmptySpace>
</div> </div>
) )

View File

@ -153,126 +153,128 @@ export const SingleModuleCard: React.FC<Props> = ({ module, handleEditModule })
setIsOpen={setModuleDeleteModal} setIsOpen={setModuleDeleteModal}
data={module} data={module}
/> />
<div className="h-full w-full min-w-[360px]"> <div className="flex flex-col rounded-[10px] border bg-white text-xs">
<div className="rounded-[10px] border bg-white text-xs"> <div className="p-4">
<div className="p-4"> <div className="flex w-full flex-col gap-5">
<div className="flex w-full flex-col gap-5"> <div className="flex items-start justify-between gap-2">
<div className="flex items-start justify-between gap-2"> <Tooltip tooltipContent={module.name} position="top-left">
<Tooltip tooltipContent={module.name} position="top-left"> <Link href={`/${workspaceSlug}/projects/${module.project}/modules/${module.id}`}>
<Link href={`/${workspaceSlug}/projects/${module.project}/modules/${module.id}`}> <a className="w-full">
<a className="w-full"> <h3 className="break-all text-lg font-semibold text-black">
<h3 className="break-all text-lg font-semibold text-black"> {truncateText(module.name, 75)}
{truncateText(module.name, 75)} </h3>
</h3> </a>
</a> </Link>
</Link> </Tooltip>
</Tooltip>
<div className="flex items-center gap-1"> <div className="flex items-center gap-1">
<div className="mr-2 rounded bg-gray-100 px-2.5 py-2"> <div className="mr-2 rounded bg-gray-100 px-2.5 py-2">
<span className="capitalize">{module?.status?.replace("-", " ")}</span> <span className="capitalize">{module?.status?.replace("-", " ")}</span>
</div> </div>
{module.is_favorite ? ( {module.is_favorite ? (
<button onClick={handleRemoveFromFavorites}> <button onClick={handleRemoveFromFavorites}>
<StarIcon className="h-4 w-4 text-orange-400" fill="#f6ad55" /> <StarIcon className="h-4 w-4 text-orange-400" fill="#f6ad55" />
</button> </button>
) : (
<button onClick={handleAddToFavorites}>
<StarIcon className="h-4 w-4 " color="#858E96" />
</button>
)}
<CustomMenu width="auto" verticalEllipsis>
<CustomMenu.MenuItem onClick={handleEditModule}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<PencilIcon className="h-4 w-4" />
<span>Edit Module</span>
</span>
</CustomMenu.MenuItem>
<CustomMenu.MenuItem onClick={handleDeleteModule}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<TrashIcon className="h-4 w-4" />
<span>Delete Module</span>
</span>
</CustomMenu.MenuItem>
<CustomMenu.MenuItem onClick={handleCopyText}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<DocumentDuplicateIcon className="h-4 w-4" />
<span>Copy Module Link</span>
</span>
</CustomMenu.MenuItem>
</CustomMenu>
</div>
</div>
<div className="grid grid-cols-2 gap-4">
<div className="flex items-start gap-1">
<CalendarDaysIcon className="h-4 w-4 text-gray-900" />
<span className="text-gray-400">Start:</span>
<span>{renderShortDateWithYearFormat(startDate)}</span>
</div>
<div className="flex items-start gap-1">
<CalendarDaysIcon className="h-4 w-4 text-gray-900" />
<span className="text-gray-400">End:</span>
<span>{renderShortDateWithYearFormat(endDate)}</span>
</div>
<div className="flex items-center gap-1.5">
<UserCircleIcon className="h-4 w-4 text-gray-900" />
<span className="text-gray-400">Lead:</span>
<div>
{module.lead_detail ? (
<div className="flex items-center gap-1">
<Avatar user={module.lead_detail} />
<span>{module.lead_detail.first_name}</span>
</div>
) : ( ) : (
<button onClick={handleAddToFavorites}> <div className="flex items-center gap-1">
<StarIcon className="h-4 w-4 " color="#858E96" /> <Image
</button> src={User}
height="12px"
width="12px"
className="rounded-full"
alt="N/A"
/>
<span>N/A</span>
</div>
)} )}
<CustomMenu width="auto" verticalEllipsis>
<CustomMenu.MenuItem onClick={handleEditModule}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<PencilIcon className="h-4 w-4" />
<span>Edit Module</span>
</span>
</CustomMenu.MenuItem>
<CustomMenu.MenuItem onClick={handleDeleteModule}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<TrashIcon className="h-4 w-4" />
<span>Delete Module</span>
</span>
</CustomMenu.MenuItem>
<CustomMenu.MenuItem onClick={handleCopyText}>
<span className="flex items-center justify-start gap-2 text-gray-800">
<DocumentDuplicateIcon className="h-4 w-4" />
<span>Copy Module Link</span>
</span>
</CustomMenu.MenuItem>
</CustomMenu>
</div> </div>
</div> </div>
<div className="grid grid-cols-2 gap-4"> <div className="flex items-center gap-1.5">
<div className="flex items-start gap-1"> <UserGroupIcon className="h-4 w-4 text-gray-900" />
<CalendarDaysIcon className="h-4 w-4 text-gray-900" /> <span className="text-gray-400">Members:</span>
<span className="text-gray-400">Start:</span> <div className="flex items-center gap-1 text-xs">
<span>{renderShortDateWithYearFormat(startDate)}</span> {module.members && module.members.length > 0 ? (
</div> <AssigneesList userIds={module.members} length={3} />
<div className="flex items-start gap-1"> ) : (
<CalendarDaysIcon className="h-4 w-4 text-gray-900" /> <div className="flex items-center gap-1">
<span className="text-gray-400">End:</span> <Image
<span>{renderShortDateWithYearFormat(endDate)}</span> src={User}
</div> height="16px"
<div className="flex items-center gap-1.5"> width="16px"
<UserCircleIcon className="h-4 w-4 text-gray-900" /> className="rounded-full"
<span className="text-gray-400">Lead:</span> alt="N/A"
<div> />
{module.lead_detail ? ( <span>N/A</span>
<div className="flex items-center gap-1"> </div>
<Avatar user={module.lead_detail} /> )}
<span>{module.lead_detail.first_name}</span>
</div>
) : (
<div className="flex items-center gap-1">
<Image
src={User}
height="12px"
width="12px"
className="rounded-full"
alt="N/A"
/>
<span>N/A</span>
</div>
)}
</div>
</div>
<div className="flex items-center gap-1.5">
<UserGroupIcon className="h-4 w-4 text-gray-900" />
<span className="text-gray-400">Members:</span>
<div className="flex items-center gap-1 text-xs">
{module.members && module.members.length > 0 ? (
<AssigneesList userIds={module.members} length={3} />
) : (
<div className="flex items-center gap-1">
<Image
src={User}
height="16px"
width="16px"
className="rounded-full"
alt="N/A"
/>
<span>N/A</span>
</div>
)}
</div>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div className="flex gap-2 bg-gray-100 p-4"> </div>
<span> <div className="flex h-full items-end">
Progress <div className="flex w-full items-center gap-2 justify-self-end bg-gray-100 p-4">
<br /> <span>Progress</span>
{isNaN(completionPercentage) ? 0 : completionPercentage.toFixed(0)}% <Tooltip
</span> tooltipContent={`Progress ${
<div className="bar relative mt-1 h-1 w-full rounded bg-gray-300"> isNaN(completionPercentage) ? 0 : completionPercentage.toFixed(0)
<div }%`}
className="absolute top-0 left-0 h-1 rounded bg-green-500 duration-300" >
style={{ width: `${(completedIssues / (moduleIssues ?? []).length) * 100}%` }} <div className="bar relative h-1 w-full rounded bg-gray-300">
/> <div
</div> className="absolute top-0 left-0 h-1 rounded bg-green-500 duration-300"
style={{ width: `${(completedIssues / (moduleIssues ?? []).length) * 100}%` }}
/>
</div>
</Tooltip>
</div> </div>
</div> </div>
</div> </div>

View File

@ -144,7 +144,7 @@ export const ProjectSidebarList: FC = () => {
/> />
<div className="mt-2.5 h-full overflow-y-auto border-t bg-white pt-2.5"> <div className="mt-2.5 h-full overflow-y-auto border-t bg-white pt-2.5">
{favoriteProjects && favoriteProjects.length > 0 && ( {favoriteProjects && favoriteProjects.length > 0 && (
<div className="mt-3 flex flex-col space-y-2 px-6"> <div className="mt-3 flex flex-col space-y-2 px-3">
{!sidebarCollapse && <h5 className="text-sm font-semibold text-gray-400">Favorites</h5>} {!sidebarCollapse && <h5 className="text-sm font-semibold text-gray-400">Favorites</h5>}
{favoriteProjects.map((favoriteProject) => { {favoriteProjects.map((favoriteProject) => {
const project = favoriteProject.project_detail; const project = favoriteProject.project_detail;

View File

@ -146,7 +146,7 @@ export const SingleSidebarProject: React.FC<Props> = ({
return ( return (
<Link key={item.name} href={item.href}> <Link key={item.name} href={item.href}>
<a <a
className={`group flex items-center rounded-md px-2 py-2 text-xs font-medium outline-none ${ className={`group flex items-center rounded-md p-2 text-xs font-medium outline-none ${
item.href === router.asPath item.href === router.asPath
? "bg-indigo-50 text-gray-900" ? "bg-indigo-50 text-gray-900"
: "text-gray-500 hover:bg-indigo-50 hover:text-gray-900 focus:bg-indigo-50 focus:text-gray-900" : "text-gray-500 hover:bg-indigo-50 hover:text-gray-900 focus:bg-indigo-50 focus:text-gray-900"

View File

@ -134,7 +134,9 @@ export const SingleState: React.FC<Props> = ({
}; };
return ( return (
<div className={`group flex items-center justify-between gap-2 rounded-[10px] bg-white p-5`}> <div
className={`group flex items-center justify-between gap-2 bg-white p-5 first:rounded-t-[10px] last:rounded-b-[10px]`}
>
<div className="flex items-center gap-3"> <div className="flex items-center gap-3">
{getStateGroupIcon(state.group, "20", "20", state.color)} {getStateGroupIcon(state.group, "20", "20", state.color)}
<div> <div>

View File

@ -1,6 +1,8 @@
import React from "react"; import React from "react";
import Image from "next/image"; import Image from "next/image";
// ui
import { PrimaryButton } from "components/ui";
// icon // icon
import { PlusIcon } from "@heroicons/react/24/outline"; import { PlusIcon } from "@heroicons/react/24/outline";
// helper // helper
@ -47,9 +49,8 @@ export const EmptyState: React.FC<Props> = ({ type, title, description, imgURL,
)} )}
<p className="max-w-md text-sm text-gray-500">{description}</p> <p className="max-w-md text-sm text-gray-500">{description}</p>
<button <PrimaryButton
type="button" className="flex items-center gap-1"
className="flex items-center gap-1 rounded-lg bg-theme px-2.5 py-2 text-sm text-white"
onClick={() => { onClick={() => {
if (action) { if (action) {
action(); action();
@ -66,7 +67,7 @@ export const EmptyState: React.FC<Props> = ({ type, title, description, imgURL,
> >
<PlusIcon className="h-4 w-4 font-bold text-white" /> <PlusIcon className="h-4 w-4 font-bold text-white" />
Create New {capitalizeFirstLetter(type)} Create New {capitalizeFirstLetter(type)}
</button> </PrimaryButton>
</div> </div>
); );
}; };

View File

@ -66,15 +66,11 @@ export const MultiLevelDropdown: React.FC<MultiLevelDropdownProps> = ({
as="button" as="button"
onClick={(e: any) => { onClick={(e: any) => {
if (option.children) { if (option.children) {
if (openChildFor === option.id) { e.stopPropagation();
e.stopPropagation(); e.preventDefault();
e.preventDefault();
setOpenChildFor(null); if (openChildFor === option.id) setOpenChildFor(null);
} else { else setOpenChildFor(option.id);
e.stopPropagation();
e.preventDefault();
setOpenChildFor(option.id);
}
} else { } else {
onSelect(option.value); onSelect(option.value);
} }

View File

@ -74,13 +74,15 @@ export const WorkspaceSidebarDropdown = () => {
return ( return (
<div className="relative"> <div className="relative">
<Menu as="div" className="col-span-4 inline-block w-full px-5 py-3 text-left"> <Menu as="div" className="col-span-4 inline-block w-full p-3 text-left">
<div className="flex w-full items-center justify-between gap-2.5"> <div className="flex items-center justify-between gap-2.5">
<Menu.Button <Menu.Button className="flex w-full items-center rounded-md py-2 text-sm font-semibold text-gray-700 focus:outline-none">
className={`inline-flex w-full items-center rounded-md px-1 py-2 text-sm font-semibold text-gray-700 focus:outline-none `} <div
> className={`flex w-full items-center gap-x-2 rounded-md bg-gray-100 px-2 py-1.5 ${
<div className="flex w-full items-center gap-x-2 rounded-md bg-gray-100 px-2 py-1.5"> sidebarCollapse ? "justify-center" : ""
<div className="relative flex h-6 w-6 items-center justify-center rounded bg-gray-700 p-2 uppercase text-white"> }`}
>
<div className="relative grid h-6 w-6 place-items-center rounded bg-gray-700 uppercase text-white">
{activeWorkspace?.logo && activeWorkspace.logo !== "" ? ( {activeWorkspace?.logo && activeWorkspace.logo !== "" ? (
<Image <Image
src={activeWorkspace.logo} src={activeWorkspace.logo}
@ -95,12 +97,8 @@ export const WorkspaceSidebarDropdown = () => {
</div> </div>
{!sidebarCollapse && ( {!sidebarCollapse && (
<p className="text-base"> <p>
{activeWorkspace?.name {activeWorkspace?.name ? truncateText(activeWorkspace.name, 14) : "Loading..."}
? activeWorkspace.name.length > 17
? `${activeWorkspace.name.substring(0, 15)}...`
: activeWorkspace.name
: "Loading..."}
</p> </p>
)} )}
</div> </div>
@ -130,12 +128,9 @@ export const WorkspaceSidebarDropdown = () => {
className="fixed left-2 z-20 mt-1 flex w-full max-w-[17rem] origin-top-left flex-col rounded-md className="fixed left-2 z-20 mt-1 flex w-full max-w-[17rem] origin-top-left flex-col rounded-md
bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none" bg-white shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none"
> >
<div className="flex flex-col items-start justify-start gap-3 px-5 py-3"> <div className="flex flex-col items-start justify-start gap-3 p-3">
<Menu.Item as="div" className="text-sm text-gray-500"> <div className="text-sm text-gray-500">{user?.email}</div>
{user?.email}
</Menu.Item>
<span className="text-sm font-semibold text-gray-500">Workspace</span> <span className="text-sm font-semibold text-gray-500">Workspace</span>
{workspaces ? ( {workspaces ? (
<div className="flex h-full w-full flex-col items-start justify-start gap-3.5"> <div className="flex h-full w-full flex-col items-start justify-start gap-3.5">
{workspaces.length > 0 ? ( {workspaces.length > 0 ? (

View File

@ -37,7 +37,7 @@ export const WorkspaceSidebarMenu: React.FC = () => {
const { collapsed: sidebarCollapse } = useTheme(); const { collapsed: sidebarCollapse } = useTheme();
return ( return (
<div className="flex w-full flex-col items-start justify-start gap-2 px-5 py-1"> <div className="flex w-full flex-col items-start justify-start gap-2 px-3 py-1">
{workspaceLinks(workspaceSlug as string).map((link, index) => ( {workspaceLinks(workspaceSlug as string).map((link, index) => (
<Link key={index} href={link.href}> <Link key={index} href={link.href}>
<a <a
@ -49,12 +49,16 @@ export const WorkspaceSidebarMenu: React.FC = () => {
sidebarCollapse ? "justify-center" : "" sidebarCollapse ? "justify-center" : ""
}`} }`}
> >
<link.icon <span className="grid h-5 w-5 flex-shrink-0 place-items-center">
className={`${ <link.icon
link.href === router.asPath ? "text-gray-900" : "text-gray-600" className={`${
} h-5 w-5 flex-shrink-0 group-hover:text-gray-900`} link.href === router.asPath ? "text-gray-900" : "text-gray-600"
aria-hidden="true" } group-hover:text-gray-900`}
/> aria-hidden="true"
height="20"
width="20"
/>
</span>
{!sidebarCollapse && link.name} {!sidebarCollapse && link.name}
</a> </a>
</Link> </Link>

View File

@ -20,8 +20,8 @@ const Sidebar: React.FC<SidebarProps> = ({ toggleSidebar, setToggleSidebar }) =>
return ( return (
<nav className="relative z-20 h-screen"> <nav className="relative z-20 h-screen">
<div <div
className={`${sidebarCollapse ? "" : "w-auto md:w-72"} fixed inset-y-0 top-0 ${ className={`${sidebarCollapse ? "" : "w-auto md:w-[17rem]"} fixed inset-y-0 top-0 ${
toggleSidebar ? "left-0" : "-left-60 md:left-0" toggleSidebar ? "left-0" : "-left-full md:left-0"
} flex h-full flex-col bg-white duration-300 md:relative`} } flex h-full flex-col bg-white duration-300 md:relative`}
> >
<div className="flex h-full flex-1 flex-col border-r"> <div className="flex h-full flex-1 flex-col border-r">

View File

@ -32,6 +32,7 @@ import {
// types // types
import type { NextPage, GetServerSidePropsContext } from "next"; import type { NextPage, GetServerSidePropsContext } from "next";
import type { IUser } from "types"; import type { IUser } from "types";
import { useRouter } from "next/dist/client/router";
const defaultValues: Partial<IUser> = { const defaultValues: Partial<IUser> = {
avatar: "", avatar: "",
@ -45,6 +46,9 @@ const Profile: NextPage = () => {
const [isRemoving, setIsRemoving] = useState(false); const [isRemoving, setIsRemoving] = useState(false);
const [isImageUploadModalOpen, setIsImageUploadModalOpen] = useState(false); const [isImageUploadModalOpen, setIsImageUploadModalOpen] = useState(false);
const router = useRouter();
const { workspaceSlug } = router.query;
const { const {
register, register,
handleSubmit, handleSubmit,
@ -131,7 +135,7 @@ const Profile: NextPage = () => {
title: "Assigned Issues", title: "Assigned Issues",
number: assignedIssuesLength, number: assignedIssuesLength,
description: "View your workspace invitations.", description: "View your workspace invitations.",
href: "/invitations", href: `/${workspaceSlug}/me/my-issues`,
}, },
{ {
icon: UserPlusIcon, icon: UserPlusIcon,

View File

@ -82,10 +82,10 @@ const ProjectViews: NextPage = () => {
/> />
{views ? ( {views ? (
views.length > 0 ? ( views.length > 0 ? (
<div className="rounded-md border"> <div className="rounded-[10px] border">
{views.map((view) => ( {views.map((view) => (
<div <div
className="flex items-center justify-between rounded-md border-b bg-white p-4" className="flex items-center justify-between border-b bg-white p-4 first:rounded-t-[10px] last:rounded-b-[10px]"
key={view.id} key={view.id}
> >
<Link href={`/${workspaceSlug}/projects/${projectId}/views/${view.id}`}> <Link href={`/${workspaceSlug}/projects/${projectId}/views/${view.id}`}>
@ -110,7 +110,7 @@ const ProjectViews: NextPage = () => {
<EmptyState <EmptyState
type="view" type="view"
title="Create New View" title="Create New View"
description="Views are smaller, focused projects that help you group and organize issues within a specific time frame." description="Views aid in saving your issues by applying various filters and grouping options."
imgURL={emptyView} imgURL={emptyView}
action={() => setIsCreateViewModalOpen(true)} action={() => setIsCreateViewModalOpen(true)}
/> />

File diff suppressed because one or more lines are too long

Before

Width:  |  Height:  |  Size: 19 KiB

After

Width:  |  Height:  |  Size: 46 KiB