replaced useTuitioDecodedToken with useTuitioUserInfo
parent
c4275df4bc
commit
9c80ed0bfb
|
@ -0,0 +1,45 @@
|
||||||
|
import React from "react";
|
||||||
|
import PropTypes from "prop-types";
|
||||||
|
import {
|
||||||
|
ListItem,
|
||||||
|
ListItemText,
|
||||||
|
ListItemIcon,
|
||||||
|
Link,
|
||||||
|
Tooltip
|
||||||
|
} from "@material-ui/core";
|
||||||
|
|
||||||
|
const ContactOption = ({ tooltip, label, link, onClick, ...props }) => {
|
||||||
|
const linkLabel = label ?? link;
|
||||||
|
return (
|
||||||
|
<ListItem dense>
|
||||||
|
<ListItemIcon>
|
||||||
|
<props.icon />
|
||||||
|
</ListItemIcon>
|
||||||
|
<ListItemText
|
||||||
|
primary={
|
||||||
|
<Tooltip title={tooltip}>
|
||||||
|
{onClick ? (
|
||||||
|
<Link href="#" onClick={onClick}>
|
||||||
|
{linkLabel}
|
||||||
|
</Link>
|
||||||
|
) : (
|
||||||
|
<Link href={link} target="_blank">
|
||||||
|
{linkLabel}
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
ContactOption.propTypes = {
|
||||||
|
icon: PropTypes.object.isRequired,
|
||||||
|
tooltip: PropTypes.string.isRequired,
|
||||||
|
label: PropTypes.string,
|
||||||
|
link: PropTypes.string.isRequired,
|
||||||
|
onClick: PropTypes.func
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ContactOption;
|
|
@ -0,0 +1,148 @@
|
||||||
|
import React, { useCallback, useMemo } from "react";
|
||||||
|
import PropTypes from "prop-types";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import ContactOption from "./ContactOption";
|
||||||
|
import BusinessCenterIcon from "@material-ui/icons/BusinessCenter";
|
||||||
|
import EmailIcon from "@material-ui/icons/Email";
|
||||||
|
import PhoneAndroidIcon from "@material-ui/icons/PhoneAndroid";
|
||||||
|
import LanguageIcon from "@material-ui/icons/Language";
|
||||||
|
import LinkedInIcon from "@material-ui/icons/LinkedIn";
|
||||||
|
import GitHubIcon from "@material-ui/icons/GitHub";
|
||||||
|
import RedditIcon from "@material-ui/icons/Reddit";
|
||||||
|
import BookIcon from "@material-ui/icons/Book";
|
||||||
|
import MenuBookIcon from "@material-ui/icons/MenuBook";
|
||||||
|
import { useClipboard } from "../../../../hooks";
|
||||||
|
|
||||||
|
const icons = {
|
||||||
|
EMAIL: EmailIcon,
|
||||||
|
PHONE: PhoneAndroidIcon,
|
||||||
|
PORTFOLIO: BusinessCenterIcon,
|
||||||
|
LINKEDIN: LinkedInIcon,
|
||||||
|
GITHUB: GitHubIcon,
|
||||||
|
GITEA: GitHubIcon,
|
||||||
|
REDDIT: RedditIcon,
|
||||||
|
BLOG: BookIcon,
|
||||||
|
CURRICULUM_VITAE: MenuBookIcon,
|
||||||
|
DEFAULT: LanguageIcon
|
||||||
|
};
|
||||||
|
|
||||||
|
const tooltips = {
|
||||||
|
EMAIL: "Generic.SendEmail",
|
||||||
|
PHONE: "Generic.Copy",
|
||||||
|
PORTFOLIO: "User.Profile.OpenPortfolio",
|
||||||
|
DEFAULT: "Generic.OpenInNewTab"
|
||||||
|
};
|
||||||
|
|
||||||
|
const orderNumbers = {
|
||||||
|
PORTFOLIO: 1,
|
||||||
|
EMAIL: 2,
|
||||||
|
PHONE: 3,
|
||||||
|
CURRICULUM_VITAE: 4,
|
||||||
|
LINKEDIN: 5,
|
||||||
|
GITEA: 6,
|
||||||
|
GITHUB: 7,
|
||||||
|
BLOG: 8,
|
||||||
|
WEBSITE: 9,
|
||||||
|
REDDIT: 10,
|
||||||
|
DEFAULT: 100
|
||||||
|
};
|
||||||
|
|
||||||
|
const getIcon = contactOption => {
|
||||||
|
const icon = icons[contactOption.contactTypeCode] || icons.DEFAULT;
|
||||||
|
return icon;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getTooltip = contactOption => {
|
||||||
|
const tooltip = tooltips[contactOption.contactTypeCode] || tooltips.DEFAULT;
|
||||||
|
return tooltip;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getOrderNumber = contactOption => {
|
||||||
|
const orderNo =
|
||||||
|
orderNumbers[contactOption.contactTypeCode] || orderNumbers.DEFAULT;
|
||||||
|
return orderNo;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLabel = (contactOption, userName) => {
|
||||||
|
switch (contactOption.contactTypeCode) {
|
||||||
|
case "EMAIL":
|
||||||
|
case "PHONE":
|
||||||
|
return contactOption.contactValue;
|
||||||
|
case "PORTFOLIO":
|
||||||
|
return userName;
|
||||||
|
default:
|
||||||
|
return contactOption.contactTypeName;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleEmailSending = email => event => {
|
||||||
|
window.location.href = `mailto:${email}`;
|
||||||
|
event.preventDefault();
|
||||||
|
};
|
||||||
|
|
||||||
|
const ContactOptions = ({ contactOptions, userName }) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { copy } = useClipboard();
|
||||||
|
|
||||||
|
const getOnClickEvent = useCallback(
|
||||||
|
contactOption => {
|
||||||
|
switch (contactOption.contactTypeCode) {
|
||||||
|
case "EMAIL":
|
||||||
|
return handleEmailSending(contactOption.contactValue);
|
||||||
|
case "PHONE":
|
||||||
|
return copy(contactOption.contactValue);
|
||||||
|
default:
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[copy]
|
||||||
|
);
|
||||||
|
|
||||||
|
const enrichedContactOptions = useMemo(
|
||||||
|
() =>
|
||||||
|
contactOptions.map(co => {
|
||||||
|
const icon = getIcon(co);
|
||||||
|
const tooltip = getTooltip(co);
|
||||||
|
const label = getLabel(co, userName);
|
||||||
|
const onClick = getOnClickEvent(co);
|
||||||
|
const orderNo = getOrderNumber(co);
|
||||||
|
const option = {
|
||||||
|
icon,
|
||||||
|
tooltip: t(tooltip),
|
||||||
|
label,
|
||||||
|
link: co.contactValue,
|
||||||
|
onClick,
|
||||||
|
orderNo
|
||||||
|
};
|
||||||
|
return option;
|
||||||
|
}),
|
||||||
|
[contactOptions, getOnClickEvent, t, userName]
|
||||||
|
);
|
||||||
|
|
||||||
|
const sorted = useMemo(
|
||||||
|
() => enrichedContactOptions.sort((a, b) => a.orderNo - b.orderNo),
|
||||||
|
[enrichedContactOptions]
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{sorted.map((z, index) => (
|
||||||
|
<ContactOption
|
||||||
|
key={`contact_${index}`}
|
||||||
|
icon={z.icon}
|
||||||
|
tooltip={z.tooltip}
|
||||||
|
label={z.label}
|
||||||
|
link={z.link}
|
||||||
|
onClick={z.onClick}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
ContactOptions.propTypes = {
|
||||||
|
contactOptions: PropTypes.array,
|
||||||
|
userName: PropTypes.object.isRequired
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ContactOptions;
|
|
@ -11,10 +11,9 @@ import {
|
||||||
Tooltip
|
Tooltip
|
||||||
} from "@material-ui/core";
|
} from "@material-ui/core";
|
||||||
import UserProfilePicture from "./UserProfilePicture";
|
import UserProfilePicture from "./UserProfilePicture";
|
||||||
import BusinessCenterIcon from "@material-ui/icons/BusinessCenter";
|
|
||||||
import { FileCopyOutlined } from "@material-ui/icons";
|
import { FileCopyOutlined } from "@material-ui/icons";
|
||||||
import EmailIcon from "@material-ui/icons/Email";
|
import ContactOptions from "./ContactOptions";
|
||||||
import { useToast } from "../../../../hooks";
|
import { useClipboard } from "../../../../hooks";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { makeStyles } from "@material-ui/core/styles";
|
import { makeStyles } from "@material-ui/core/styles";
|
||||||
import styles from "../styles";
|
import styles from "../styles";
|
||||||
|
@ -22,65 +21,28 @@ import styles from "../styles";
|
||||||
const useStyles = makeStyles(styles);
|
const useStyles = makeStyles(styles);
|
||||||
|
|
||||||
const UserProfileCardContent = ({ userData }) => {
|
const UserProfileCardContent = ({ userData }) => {
|
||||||
const { email, profilePictureUrl } = userData;
|
const { profilePictureUrl } = userData;
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { info } = useToast();
|
|
||||||
const classes = useStyles();
|
const classes = useStyles();
|
||||||
|
const { copy } = useClipboard();
|
||||||
const handleCopyToClipboard = url => () => {
|
|
||||||
navigator.clipboard.writeText(url);
|
|
||||||
info(t("Generic.CopiedToClipboard"));
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleEmailSending = event => {
|
|
||||||
window.location.href = `mailto:${email}`;
|
|
||||||
event.preventDefault();
|
|
||||||
};
|
|
||||||
|
|
||||||
const userName = `${userData.firstName} ${userData.lastName}`;
|
const userName = `${userData.firstName} ${userData.lastName}`;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={classes.panel}>
|
<div className={classes.panel}>
|
||||||
<UserProfilePicture userData={userData} />
|
<UserProfilePicture pictureUrl={userData.profilePictureUrl} />
|
||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid item xs={12} sm={6}>
|
<Grid item xs={12} sm={6}>
|
||||||
<List>
|
<List>
|
||||||
<ListItem dense>
|
<ContactOptions
|
||||||
<ListItemIcon>
|
contactOptions={userData.contactOptions}
|
||||||
<BusinessCenterIcon />
|
userName={userName}
|
||||||
</ListItemIcon>
|
|
||||||
<ListItemText
|
|
||||||
primary={
|
|
||||||
<Tooltip title={t("User.Profile.OpenPortfolio")}>
|
|
||||||
<Link href="https://lab.code-rove.com/tsp/" target="_blank">
|
|
||||||
{userName}
|
|
||||||
</Link>
|
|
||||||
</Tooltip>
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
|
||||||
<ListItem dense>
|
|
||||||
<ListItemIcon>
|
|
||||||
<EmailIcon />
|
|
||||||
</ListItemIcon>
|
|
||||||
<ListItemText
|
|
||||||
primary={
|
|
||||||
<Tooltip title={t("Generic.SendEmail")}>
|
|
||||||
<Link href="#" onClick={handleEmailSending}>
|
|
||||||
{email}
|
|
||||||
</Link>
|
|
||||||
</Tooltip>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ListItem>
|
|
||||||
{profilePictureUrl && (
|
{profilePictureUrl && (
|
||||||
<ListItem dense>
|
<ListItem dense>
|
||||||
<ListItemIcon>
|
<ListItemIcon>
|
||||||
<Tooltip title={t("Generic.Copy")}>
|
<Tooltip title={t("Generic.Copy")}>
|
||||||
<IconButton
|
<IconButton size="small" onClick={copy(profilePictureUrl)}>
|
||||||
size="small"
|
|
||||||
onClick={handleCopyToClipboard(profilePictureUrl)}
|
|
||||||
>
|
|
||||||
<FileCopyOutlined />
|
<FileCopyOutlined />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
|
@ -11,9 +11,9 @@ const UserProfileComponent = ({ userData }) => {
|
||||||
const userLoginDate = useMemo(
|
const userLoginDate = useMemo(
|
||||||
() =>
|
() =>
|
||||||
t("DATE_FORMAT", {
|
t("DATE_FORMAT", {
|
||||||
date: { value: userData.createdAt, format: "DD-MM-YYYY HH:mm:ss" }
|
date: { value: userData.lastLoginDate, format: "DD-MM-YYYY HH:mm:ss" }
|
||||||
}),
|
}),
|
||||||
[t, userData.createdAt]
|
[t, userData.lastLoginDate]
|
||||||
);
|
);
|
||||||
|
|
||||||
const userDescription = t("User.Profile.Description", {
|
const userDescription = t("User.Profile.Description", {
|
||||||
|
|
|
@ -1,12 +1,10 @@
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { useTuitioDecodedToken } from "@flare/tuitio-client-react";
|
import { useTuitioUserInfo } from "@flare/tuitio-client-react";
|
||||||
import UserProfileComponent from "./UserProfileComponent";
|
import UserProfileComponent from "./UserProfileComponent";
|
||||||
|
|
||||||
const UserProfileContainer = () => {
|
const UserProfileContainer = () => {
|
||||||
const { decodedToken } = useTuitioDecodedToken();
|
const { userInfo } = useTuitioUserInfo();
|
||||||
return (
|
return <>{userInfo && <UserProfileComponent userData={userInfo} />}</>;
|
||||||
<>{decodedToken && <UserProfileComponent userData={decodedToken} />}</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default UserProfileContainer;
|
export default UserProfileContainer;
|
||||||
|
|
|
@ -7,15 +7,14 @@ import DefaultUserProfilePicture from "../../../../assets/images/DefaultUserProf
|
||||||
|
|
||||||
const useStyles = makeStyles(style);
|
const useStyles = makeStyles(style);
|
||||||
|
|
||||||
const UserProfilePicture = ({ userData }) => {
|
const UserProfilePicture = ({ pictureUrl }) => {
|
||||||
const classes = useStyles();
|
const classes = useStyles();
|
||||||
const { profilePictureUrl } = userData;
|
const url = pictureUrl ?? DefaultUserProfilePicture;
|
||||||
const url = profilePictureUrl ?? DefaultUserProfilePicture;
|
|
||||||
return <Avatar src={url} alt="..." className={classes.profilePicture} />;
|
return <Avatar src={url} alt="..." className={classes.profilePicture} />;
|
||||||
};
|
};
|
||||||
|
|
||||||
UserProfilePicture.propTypes = {
|
UserProfilePicture.propTypes = {
|
||||||
userData: PropTypes.object.isRequired
|
pictureUrl: PropTypes.string
|
||||||
};
|
};
|
||||||
|
|
||||||
export default UserProfilePicture;
|
export default UserProfilePicture;
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import { useToast } from "./useToast";
|
import { useToast } from "./useToast";
|
||||||
import { useSensitiveInfo } from "../providers/SensitiveInfoProvider";
|
import { useSensitiveInfo } from "../providers/SensitiveInfoProvider";
|
||||||
|
import { useClipboard } from "./useClipboard";
|
||||||
|
|
||||||
export { useToast, useSensitiveInfo };
|
export { useToast, useSensitiveInfo, useClipboard };
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
import { useCallback } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useToast } from "./useToast";
|
||||||
|
|
||||||
|
const useClipboard = () => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { info } = useToast();
|
||||||
|
const copy = useCallback(
|
||||||
|
url => () => {
|
||||||
|
navigator.clipboard.writeText(url);
|
||||||
|
info(t("Generic.CopiedToClipboard"));
|
||||||
|
},
|
||||||
|
[info, t]
|
||||||
|
);
|
||||||
|
return { copy };
|
||||||
|
};
|
||||||
|
|
||||||
|
export { useClipboard };
|
Loading…
Reference in New Issue