Skip to content

Commit

Permalink
feat: add user page design
Browse files Browse the repository at this point in the history
  • Loading branch information
andre-code committed Dec 10, 2024
1 parent 6676127 commit 5376abb
Show file tree
Hide file tree
Showing 4 changed files with 120 additions and 94 deletions.
5 changes: 2 additions & 3 deletions client/src/components/EntityWatermark.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,10 @@

import cx from "classnames";
import { CSSProperties } from "react";
import { Folder, People, Person } from "react-bootstrap-icons";
import { People, Person } from "react-bootstrap-icons";

interface EntityWatermarkProps {
type: "project" | "user" | "group";
type: "user" | "group";
}
export function EntityWatermark({ type }: EntityWatermarkProps) {
const watermarkStyles: CSSProperties = {
Expand All @@ -37,7 +37,6 @@ export function EntityWatermark({ type }: EntityWatermarkProps) {
>
{type === "group" && <People />}
{type === "user" && <Person />}
{type === "project" && <Folder />}
</div>
);
}
4 changes: 1 addition & 3 deletions client/src/features/projectsV2/list/ProjectV2ListDisplay.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -122,9 +122,7 @@ export default function ProjectListDisplay({
</div>
</div>
)}
{!data.total && emptyListElement && (
<p>This group doesnt has projects yet.</p>
)}
{!data.total && emptyListElement}
{data.projects.length > 0 && (
<>
<div className={cx("d-flex", "flex-column", "gap-3")}>
Expand Down
23 changes: 0 additions & 23 deletions client/src/features/usersV2/show/UserAvatar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -103,26 +103,3 @@ export function AvatarTypeWrap({ type, children }: AvatarType) {
</div>
);
}

interface AvatarType {
type: "User" | "Group";
children: ReactNode;
}
export function AvatarTypeWrap({ type, children }: AvatarType) {
const styles: CSSProperties = {
width: "75px",
height: "65px",
};

return (
<div
style={styles}
className={cx("d-flex", "align-items-end", "position-relative")}
>
{children}
<div style={{ right: "0" }} className={cx("position-absolute", "top-0")}>
<EntityPill entityType={type} size="sm" />
</div>
</div>
);
}
182 changes: 117 additions & 65 deletions client/src/features/usersV2/show/UserShow.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,12 +19,14 @@
import { skipToken } from "@reduxjs/toolkit/query";
import cx from "classnames";
import { useEffect } from "react";
import { InfoCircle, JournalAlbum } from "react-bootstrap-icons";
import {
generatePath,
useNavigate,
useParams,
} from "react-router-dom-v5-compat";
import { Badge, Col, Row } from "reactstrap";
import { Badge, Card, CardBody, CardHeader, Col, Row } from "reactstrap";
import { EntityWatermark } from "../../../components/EntityWatermark.tsx";

import { Loader } from "../../../components/Loader";
import ContainerWrap from "../../../components/container/ContainerWrap";
Expand All @@ -34,13 +36,18 @@ import DataConnectorsBox from "../../dataConnectorsV2/components/DataConnectorsB
import { useGetNamespacesByNamespaceSlugQuery } from "../../projectsV2/api/projectV2.enhanced-api";
import ProjectV2ListDisplay from "../../projectsV2/list/ProjectV2ListDisplay";
import UserNotFound from "../../projectsV2/notFound/UserNotFound";
import { EntityPill } from "../../searchV2/components/SearchV2Results";
import { useGetUserByIdQuery, useGetUserQuery } from "../api/users.api";
import UserAvatar, { UserAvatarSize } from "./UserAvatar";
import {
useGetUserByIdQuery,
useGetUserQuery,
UserWithId,
} from "../api/users.api";
import UserAvatar, { AvatarTypeWrap, UserAvatarSize } from "./UserAvatar";

export default function UserShow() {
const { username } = useParams<{ username: string }>();

const { data: currentUser } = useGetUserQuery();
const isCurrentUser =
currentUser?.isLoggedIn && currentUser.username === username;
const navigate = useNavigate();

const {
Expand All @@ -50,6 +57,7 @@ export default function UserShow() {
} = useGetNamespacesByNamespaceSlugQuery(
username ? { namespaceSlug: username } : skipToken
);

const {
data: user,
isLoading: isLoadingUser,
Expand All @@ -60,6 +68,11 @@ export default function UserShow() {
: skipToken
);

const name =
user?.first_name && user?.last_name
? `${user.first_name} ${user.last_name}`
: user?.first_name || user?.last_name;

const isLoading = isLoadingNamespace || isLoadingUser;
const error = namespaceError ?? userError;

Expand All @@ -82,68 +95,76 @@ export default function UserShow() {
return <UserNotFound error={error} />;
}

const name =
user.first_name && user.last_name
? `${user.first_name} ${user.last_name}`
: user.first_name || user.last_name;

return (
<ContainerWrap>
<div className={cx("d-flex", "flex-column", "flex-sm-row", "gap-2")}>
<div>
<div
className={cx(
"d-flex",
"flex-row",
"flex-wrap",
"flex-sm-nowrap",
"gap-2"
)}
>
<div className={cx("align-items-center", "d-flex", "gap-2")}>
<UserAvatar
firstName={user.first_name}
lastName={user.last_name}
username={username}
size={UserAvatarSize.medium}
/>
<h2 className="mb-0">{name ?? "Unknown user"}</h2>
</div>

<div className={cx("align-items-center", "d-flex", "gap-2")}>
<EntityPill
entityType="User"
size="sm"
tooltipPlacement="bottom"
/>
<ItsYouBadge username={username} />
</div>
</div>
<p className="fst-italic">{`@${username}`}</p>
</div>
const information = (
<div className={cx("d-flex", "flex-column", "gap-3")}>
<div>
<p className={cx("align-items-center", "d-flex", "gap-2", "mb-0")}>
<JournalAlbum className="bi" />
Identifier:
</p>
<div className="ms-4">@{username}</div>
</div>
</div>
);

<section>
<h4>Personal Projects</h4>
<ProjectV2ListDisplay
namespace={username}
pageParam="projects_page"
emptyListElement={
<p>{name ?? username} has no visible personal projects.</p>
}
/>
</section>
<section className="mt-3">
<Row>
<Col className="order-3" xs={12} xl={8}>
<DataConnectorsBox
namespace={username}
namespaceKind="user"
pageParam="data_connectors_page"
/>
</Col>
</Row>
</section>
return (
<ContainerWrap className="position-relative">
<EntityWatermark type="user" />
<Row>
<Col xs={12} className={cx("mb-3", "pt-2", "pb-5")}>
<UserHeader user={user} username={username} name={name ?? ""} />
</Col>
<Col xs={12}>
<Row className="g-4">
<Col xs={12} md={8} xl={9}>
<Row className="g-4">
<Col xs={12}>
<ProjectV2ListDisplay
namespace={username}
pageParam="projects_page"
emptyListElement={
isCurrentUser ? (
<p>
Collaborate on projects with anyone, with data, code,
and compute together in one place.
</p>
) : (
<p>This user has no visible personal projects.</p>
)
}
/>
</Col>
<Col className="order-3" xs={12}>
<DataConnectorsBox
namespace={username}
namespaceKind="user"
pageParam="data_connectors_page"
/>
</Col>
</Row>
</Col>
<Col xs={12} md={4} xl={3}>
<Card data-cy="project-info-card">
<CardHeader>
<div
className={cx(
"align-items-center",
"d-flex",
"justify-content-between"
)}
>
<h4 className="m-0">
<InfoCircle className={cx("me-1", "bi")} />
Info
</h4>
</div>
</CardHeader>
<CardBody>{information}</CardBody>
</Card>
</Col>
</Row>
</Col>
</Row>
</ContainerWrap>
);
}
Expand Down Expand Up @@ -173,3 +194,34 @@ function ItsYouBadge({ username }: ItsYouBadgeProps) {

return null;
}

function UserHeader({
user,
username,
name,
}: {
user: UserWithId;
username: string;
name: string;
}) {
return (
<div className={cx("d-flex", "flex-row", "flex-nowrap", "gap-2")}>
<div className={cx("d-flex", "gap-2")}>
<AvatarTypeWrap type={"User"}>
<UserAvatar
firstName={user.first_name}
lastName={user.last_name}
username={username}
size={UserAvatarSize.extraLarge}
/>
</AvatarTypeWrap>
<div className="d-flex gap-2">
<h2 className="mb-0">{name}</h2>
<div>
<ItsYouBadge username={username} />
</div>
</div>
</div>
</div>
);
}

0 comments on commit 5376abb

Please sign in to comment.