mirror of
https://github.com/filecoin-project/slate.git
synced 2024-11-08 23:51:22 +03:00
feat(CollectionPreview): update hover animation to use transforms
This commit is contained in:
parent
e5fd28ac9c
commit
f1620d984d
@ -209,8 +209,8 @@ export const grids = {
|
||||
mobile: { width: 166, rowGap: 12 },
|
||||
},
|
||||
collection: {
|
||||
desktop: { width: 432, rowGap: 16 },
|
||||
mobile: { width: 300, rowGap: 8 },
|
||||
desktop: { width: 382, rowGap: 16 },
|
||||
mobile: { width: 280, rowGap: 8 },
|
||||
},
|
||||
profile: {
|
||||
desktop: { width: 248, rowGap: 16 },
|
||||
|
@ -1,22 +1,16 @@
|
||||
import * as React from "react";
|
||||
import * as Validations from "~/common/validations";
|
||||
import * as Typography from "~/components/system/components/Typography";
|
||||
import * as Styles from "~/common/styles";
|
||||
import * as Strings from "~/common/strings";
|
||||
import * as Constants from "~/common/constants";
|
||||
import * as SVG from "~/common/svg";
|
||||
|
||||
import { Logo } from "~/common/logo";
|
||||
import { useInView } from "~/common/hooks";
|
||||
import { isBlurhashValid } from "blurhash";
|
||||
import { Blurhash } from "react-blurhash";
|
||||
import { css } from "@emotion/react";
|
||||
import { FollowButton } from "~/components/core/CollectionPreviewBlock/components";
|
||||
import { useFollowHandler } from "~/components/core/CollectionPreviewBlock/hooks";
|
||||
import { Link } from "~/components/core/Link";
|
||||
import { motion } from "framer-motion";
|
||||
|
||||
import ObjectPlaceholder from "~/components/core/ObjectPreview/placeholders";
|
||||
import { Preview } from "~/components/core/CollectionPreviewBlock/components";
|
||||
import { AspectRatio } from "~/components/system";
|
||||
|
||||
const STYLES_CONTAINER = (theme) => css`
|
||||
position: relative;
|
||||
@ -27,35 +21,25 @@ const STYLES_CONTAINER = (theme) => css`
|
||||
border-radius: 16px;
|
||||
width: 100%;
|
||||
overflow: hidden;
|
||||
|
||||
height: 304px;
|
||||
@media (max-width: ${theme.sizes.mobile}px) {
|
||||
height: 281px;
|
||||
}
|
||||
`;
|
||||
|
||||
const STYLES_PREVIEW = css`
|
||||
flex-grow: 1;
|
||||
background-size: cover;
|
||||
overflow: hidden;
|
||||
img {
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
object-fit: cover;
|
||||
}
|
||||
`;
|
||||
|
||||
const STYLES_DESCRIPTION_CONTAINER = (theme) => css`
|
||||
background-color: ${theme.semantic.bgLight};
|
||||
position: absolute;
|
||||
bottom: 0%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 9px 16px 12px;
|
||||
const STYLES_DESCRIPTION = (theme) => css`
|
||||
position: relative;
|
||||
border-radius: 0px 0px 16px 16px;
|
||||
box-shadow: 0 -0.5px 0.5px ${theme.system.grayLight4};
|
||||
box-sizing: border-box;
|
||||
width: 100%;
|
||||
margin-top: auto;
|
||||
background-color: ${theme.semantic.bgLight};
|
||||
z-index: 1;
|
||||
`;
|
||||
|
||||
const STYLES_INNER_DESCRIPTION = (theme) => css`
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
background-color: ${theme.semantic.bgLight};
|
||||
padding: 9px 16px 0px;
|
||||
box-shadow: 0 -0.5px 0.5px ${theme.system.grayLight4};
|
||||
`;
|
||||
|
||||
const STYLES_SPACE_BETWEEN = css`
|
||||
@ -71,26 +55,14 @@ const STYLES_PROFILE_IMAGE = (theme) => css`
|
||||
`;
|
||||
|
||||
const STYLES_METRICS = css`
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
margin-top: auto;
|
||||
padding: 4px 16px 8px;
|
||||
${Styles.CONTAINER_CENTERED};
|
||||
${STYLES_SPACE_BETWEEN}
|
||||
`;
|
||||
|
||||
const STYLES_PLACEHOLDER_CONTAINER = css`
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
`;
|
||||
|
||||
const STYLES_EMPTY_CONTAINER = css`
|
||||
display: flex;
|
||||
overflow: hidden;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
`;
|
||||
|
||||
const STYLES_CONTROLS = css`
|
||||
position: absolute;
|
||||
z-index: 1;
|
||||
@ -104,97 +76,24 @@ const STYLES_TEXT_GRAY = (theme) => css`
|
||||
color: ${theme.semantic.textGray};
|
||||
`;
|
||||
|
||||
const getFileBlurHash = (file) => {
|
||||
const coverImage = file?.data?.coverImage;
|
||||
const coverImageBlurHash = coverImage?.data?.blurhash;
|
||||
if (coverImage && isBlurhashValid(coverImageBlurHash)) return coverImageBlurHash;
|
||||
|
||||
const blurhash = file?.data?.blurhash;
|
||||
if (isBlurhashValid(blurhash)) return blurhash;
|
||||
|
||||
return null;
|
||||
};
|
||||
|
||||
const getObjectToPreview = (objects = []) => {
|
||||
let objectIdx = 0;
|
||||
let isImage = false;
|
||||
|
||||
objects.some((object, i) => {
|
||||
const isPreviewableImage = Validations.isPreviewableImage(object.data.type);
|
||||
if (isPreviewableImage) (objectIdx = i), (isImage = true);
|
||||
return isPreviewableImage;
|
||||
});
|
||||
|
||||
return { ...objects[objectIdx], isImage };
|
||||
};
|
||||
|
||||
const Preview = ({ collection, children, ...props }) => {
|
||||
const [isLoading, setLoading] = React.useState(true);
|
||||
const handleOnLoaded = () => setLoading(false);
|
||||
|
||||
const previewerRef = React.useRef();
|
||||
const { isInView } = useInView({
|
||||
ref: previewerRef,
|
||||
});
|
||||
|
||||
const object = React.useMemo(() => getObjectToPreview(collection.objects), [collection.objects]);
|
||||
|
||||
const isCollectionEmpty = collection.objects.length === 0;
|
||||
if (isCollectionEmpty) {
|
||||
return (
|
||||
<div css={STYLES_EMPTY_CONTAINER} {...props}>
|
||||
{children}
|
||||
<Logo style={{ height: 24, marginBottom: 8, color: Constants.system.grayLight2 }} />
|
||||
<Typography.P2 color="grayLight2">This collection is empty</Typography.P2>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
if (object.isImage) {
|
||||
const { coverImage } = object.data;
|
||||
const blurhash = getFileBlurHash(object);
|
||||
const previewImage = coverImage
|
||||
? Strings.getURLfromCID(coverImage?.cid)
|
||||
: Strings.getURLfromCID(object.cid);
|
||||
|
||||
return (
|
||||
<div ref={previewerRef} css={STYLES_PREVIEW} {...props}>
|
||||
{children}
|
||||
{isInView && (
|
||||
<>
|
||||
{isLoading && blurhash && (
|
||||
<Blurhash
|
||||
hash={blurhash}
|
||||
style={{ position: "absolute", top: 0, left: 0 }}
|
||||
height="100%"
|
||||
width="100%"
|
||||
resolutionX={32}
|
||||
resolutionY={32}
|
||||
punch={1}
|
||||
/>
|
||||
)}
|
||||
<img src={previewImage} alt="Collection preview" onLoad={handleOnLoaded} />
|
||||
</>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div css={STYLES_PREVIEW} {...props}>
|
||||
{children}
|
||||
<ObjectPlaceholder ratio={1} containerCss={STYLES_PLACEHOLDER_CONTAINER} file={object} />
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
export default function CollectionPreview({ collection, viewer, owner, onAction }) {
|
||||
const [areControlsVisible, setShowControls] = React.useState(false);
|
||||
const showControls = () => setShowControls(true);
|
||||
const hideControls = () => setShowControls(false);
|
||||
|
||||
const { isDescriptionVisible, showDescription, hideDescription } = useShowDescription();
|
||||
const descriptionRef = React.useRef();
|
||||
const descriptionHeight = React.useRef();
|
||||
React.useEffect(() => {
|
||||
const element = descriptionRef.current;
|
||||
if (element) {
|
||||
descriptionHeight.current = element.offsetHeight;
|
||||
}
|
||||
}, []);
|
||||
|
||||
const description = collection?.data?.body;
|
||||
const { isDescriptionVisible, showDescription, hideDescription } = useShowDescription({
|
||||
disabled: !description,
|
||||
});
|
||||
|
||||
const { follow, followCount, isFollowed } = useFollowHandler({ collection, viewer });
|
||||
|
||||
@ -202,110 +101,125 @@ export default function CollectionPreview({ collection, viewer, owner, onAction
|
||||
|
||||
return (
|
||||
<div css={STYLES_CONTAINER}>
|
||||
<Preview collection={collection} onMouseEnter={showControls} onMouseLeave={hideControls}>
|
||||
<motion.div
|
||||
initial={{ opacity: 0 }}
|
||||
animate={{ opacity: areControlsVisible ? 1 : 0 }}
|
||||
css={STYLES_CONTROLS}
|
||||
>
|
||||
<FollowButton
|
||||
onClick={follow}
|
||||
isFollowed={isFollowed}
|
||||
followCount={followCount}
|
||||
disabled={collection.ownerId === viewer?.id}
|
||||
/>
|
||||
</motion.div>
|
||||
</Preview>
|
||||
|
||||
<div style={{ position: "relative", height: 61 }}>
|
||||
<motion.div
|
||||
css={STYLES_DESCRIPTION_CONTAINER}
|
||||
onMouseEnter={showDescription}
|
||||
onMouseLeave={hideDescription}
|
||||
transition={{ duration: 0.4, ease: "easeOut" }}
|
||||
>
|
||||
<div css={[Styles.HORIZONTAL_CONTAINER_CENTERED, STYLES_SPACE_BETWEEN]}>
|
||||
<Typography.H5 color="textBlack" nbrOflines={1} title={collection.slatename}>
|
||||
{collection.slatename}
|
||||
</Typography.H5>
|
||||
</div>
|
||||
<AspectRatio ratio={192 / 382}>
|
||||
<Preview collection={collection} onMouseEnter={showControls} onMouseLeave={hideControls}>
|
||||
<motion.div
|
||||
style={{ marginTop: 4 }}
|
||||
initial={{ height: 0 }}
|
||||
animate={{
|
||||
height: isDescriptionVisible ? 108 : 0,
|
||||
}}
|
||||
initial={{ opacity: 0 }}
|
||||
animate={{ opacity: areControlsVisible ? 1 : 0 }}
|
||||
css={STYLES_CONTROLS}
|
||||
>
|
||||
<FollowButton
|
||||
onClick={follow}
|
||||
isFollowed={isFollowed}
|
||||
followCount={followCount}
|
||||
disabled={collection.ownerId === viewer?.id}
|
||||
/>
|
||||
</motion.div>
|
||||
</Preview>
|
||||
</AspectRatio>
|
||||
|
||||
<motion.article
|
||||
css={STYLES_DESCRIPTION}
|
||||
onMouseMove={showDescription}
|
||||
onMouseLeave={hideDescription}
|
||||
>
|
||||
<div style={{ position: "relative", paddingTop: 9 }}>
|
||||
<Typography.H5 nbrOflines={1} style={{ visibility: "hidden" }}>
|
||||
{collection.slatename}
|
||||
</Typography.H5>
|
||||
|
||||
<motion.div
|
||||
css={STYLES_INNER_DESCRIPTION}
|
||||
style={{ position: "absolute", left: 0, top: 0 }}
|
||||
initial={{ y: 0 }}
|
||||
animate={{ y: isDescriptionVisible ? -descriptionHeight.current : 0 }}
|
||||
transition={{
|
||||
type: "spring",
|
||||
stiffness: 170,
|
||||
damping: 26,
|
||||
delay: isDescriptionVisible ? 0 : 0.25,
|
||||
delay: isDescriptionVisible ? 0 : 0.2,
|
||||
}}
|
||||
>
|
||||
<Typography.P2
|
||||
as={motion.p}
|
||||
initial={{ opacity: 0 }}
|
||||
animate={{
|
||||
opacity: isDescriptionVisible ? 1 : 0,
|
||||
}}
|
||||
transition={{ delay: isDescriptionVisible ? 0.25 : 0 }}
|
||||
color="textGrayDark"
|
||||
nbrOflines={5}
|
||||
>
|
||||
{description || ""}
|
||||
</Typography.P2>
|
||||
</motion.div>
|
||||
<div css={STYLES_METRICS}>
|
||||
<div css={[Styles.CONTAINER_CENTERED, STYLES_TEXT_GRAY]}>
|
||||
<SVG.Box />
|
||||
<Typography.P3 style={{ marginLeft: 4 }} color="textGray">
|
||||
{fileCount}
|
||||
</Typography.P3>
|
||||
</div>
|
||||
{owner && (
|
||||
<div style={{ alignItems: "end" }} css={Styles.CONTAINER_CENTERED}>
|
||||
<Link
|
||||
href={`/$/user/${owner.id}`}
|
||||
onAction={onAction}
|
||||
aria-label={`Visit ${owner.username}'s profile`}
|
||||
title={`Visit ${owner.username}'s profile`}
|
||||
<Typography.H5 color="textBlack" nbrOflines={1} title={collection.slatename}>
|
||||
{collection.slatename}
|
||||
</Typography.H5>
|
||||
{description && (
|
||||
<div ref={descriptionRef}>
|
||||
<Typography.P3
|
||||
as={motion.p}
|
||||
style={{ paddingTop: 3 }}
|
||||
nbrOflines={1}
|
||||
color="textGrayDark"
|
||||
initial={{ opacity: 0 }}
|
||||
animate={{ opacity: isDescriptionVisible ? 1 : 0 }}
|
||||
transition={{ delay: isDescriptionVisible ? 0.2 : 0 }}
|
||||
>
|
||||
<img
|
||||
css={STYLES_PROFILE_IMAGE}
|
||||
src={owner?.data?.photo}
|
||||
alt={`${owner.username} profile`}
|
||||
onError={(e) => (e.target.src = Constants.profileDefaultPicture)}
|
||||
/>
|
||||
</Link>
|
||||
<Link
|
||||
href={`/$/user/${owner.id}`}
|
||||
onAction={onAction}
|
||||
aria-label={`Visit ${owner.username}'s profile`}
|
||||
title={`Visit ${owner.username}'s profile`}
|
||||
>
|
||||
<Typography.P3 style={{ marginLeft: 8 }} color="textGray">
|
||||
{owner.username}
|
||||
</Typography.P3>
|
||||
</Link>
|
||||
{description}
|
||||
</Typography.P3>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</motion.div>
|
||||
</div>
|
||||
</motion.div>
|
||||
</div>
|
||||
<Metrics owner={owner} fileCount={fileCount} onAction={onAction} />
|
||||
</motion.article>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
const useShowDescription = () => {
|
||||
function Metrics({ fileCount, owner, onAction }) {
|
||||
return (
|
||||
<div css={STYLES_METRICS}>
|
||||
<div css={[Styles.CONTAINER_CENTERED, STYLES_TEXT_GRAY]}>
|
||||
<SVG.Box />
|
||||
<Typography.P3 style={{ marginLeft: 4 }} color="textGray">
|
||||
{fileCount}
|
||||
</Typography.P3>
|
||||
</div>
|
||||
{owner && (
|
||||
<div style={{ alignItems: "end" }} css={Styles.CONTAINER_CENTERED}>
|
||||
<Link
|
||||
href={`/$/user/${owner.id}`}
|
||||
onAction={onAction}
|
||||
aria-label={`Visit ${owner.username}'s profile`}
|
||||
title={`Visit ${owner.username}'s profile`}
|
||||
>
|
||||
<img
|
||||
css={STYLES_PROFILE_IMAGE}
|
||||
src={owner?.data?.photo}
|
||||
alt={`${owner.username} profile`}
|
||||
onError={(e) => (e.target.src = Constants.profileDefaultPicture)}
|
||||
/>
|
||||
</Link>
|
||||
<Link
|
||||
href={`/$/user/${owner.id}`}
|
||||
onAction={onAction}
|
||||
aria-label={`Visit ${owner.username}'s profile`}
|
||||
title={`Visit ${owner.username}'s profile`}
|
||||
>
|
||||
<Typography.P3 style={{ marginLeft: 8 }} color="textGray">
|
||||
{owner.username}
|
||||
</Typography.P3>
|
||||
</Link>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
const useShowDescription = ({ disabled }) => {
|
||||
const [isDescriptionVisible, setShowDescription] = React.useState(false);
|
||||
const timeoutId = React.useRef();
|
||||
|
||||
const showDescription = () => {
|
||||
if (disabled) return;
|
||||
|
||||
clearTimeout(timeoutId.current);
|
||||
const id = setTimeout(() => setShowDescription(true), 250);
|
||||
timeoutId.current = id;
|
||||
};
|
||||
const hideDescription = () => {
|
||||
if (disabled) return;
|
||||
|
||||
clearTimeout(timeoutId.current);
|
||||
setShowDescription(false);
|
||||
};
|
||||
|
@ -33,6 +33,10 @@ const STYLES_DESCRIPTION = (theme) => css`
|
||||
`;
|
||||
|
||||
const STYLES_INNER_DESCRIPTION = (theme) => css`
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
background-color: ${theme.semantic.bgLight};
|
||||
padding: 9px 16px 0px;
|
||||
box-shadow: 0 -0.5px 0.5px ${theme.system.grayLight4};
|
||||
@ -156,7 +160,6 @@ export default function ObjectPreviewPrimitive({
|
||||
|
||||
<motion.div
|
||||
css={STYLES_INNER_DESCRIPTION}
|
||||
style={{ position: "absolute", left: 0, top: 0 }}
|
||||
initial={{ y: 0 }}
|
||||
animate={{ y: isDescriptionVisible ? -descriptionHeight.current : 0 }}
|
||||
transition={{
|
||||
|
Loading…
Reference in New Issue
Block a user