Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: new yearn landing #571

Merged
merged 70 commits into from
Nov 20, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
70 commits
Select commit Hold shift + click to select a range
66b9786
feat: wip sidebar
Karlen9 Jul 30, 2024
8a52e0a
fix: build fix
Karlen9 Jul 30, 2024
794b952
feat: search & category sections & style changes
Karlen9 Jul 31, 2024
4dd2585
feat: carousel & search icon & hovers
Karlen9 Jul 31, 2024
2976e61
feat: cutaways + scrollbar dismiss + search layout + mobile layout
Karlen9 Aug 5, 2024
f70a5bd
feat: arrows for scroll + small fixes
Karlen9 Aug 6, 2024
1ba8a46
feat: imp of useSearch + small changes
Karlen9 Aug 7, 2024
4304c59
fix: small fix
Karlen9 Aug 7, 2024
49af5f8
feat: filterBar + sorting
Karlen9 Aug 7, 2024
13a4762
fix: small fix
Karlen9 Aug 7, 2024
d3743c7
fix: style changes
Karlen9 Aug 7, 2024
bbe5ca1
fix: mobile style fixes
Karlen9 Aug 7, 2024
9d0140a
feat: first round of apps
Majorfi Aug 7, 2024
b2ab85e
fix: type
Majorfi Aug 7, 2024
bf7d261
fix: menu
Majorfi Aug 7, 2024
9f7a2de
entrance anim
Majorfi Aug 7, 2024
b3d0db2
feat: styles updates
Karlen9 Aug 7, 2024
ac28f3c
fix: small change
Karlen9 Aug 7, 2024
48ab68b
fix: mobile add card imp
Karlen9 Aug 7, 2024
fef22cf
feat: add description to apps
Karlen9 Aug 13, 2024
681864d
feat: add icons
Karlen9 Aug 13, 2024
0a0dd0d
feat: wip sidebar
Karlen9 Jul 30, 2024
fa4a1cb
fix: build fix
Karlen9 Jul 30, 2024
1aea141
feat: search & category sections & style changes
Karlen9 Jul 31, 2024
c4781a7
feat: carousel & search icon & hovers
Karlen9 Jul 31, 2024
0c72151
feat: cutaways + scrollbar dismiss + search layout + mobile layout
Karlen9 Aug 5, 2024
3082adf
feat: arrows for scroll + small fixes
Karlen9 Aug 6, 2024
6962162
feat: imp of useSearch + small changes
Karlen9 Aug 7, 2024
494b258
fix: small fix
Karlen9 Aug 7, 2024
2701ad6
feat: filterBar + sorting
Karlen9 Aug 7, 2024
8d4050b
fix: small fix
Karlen9 Aug 7, 2024
bc15f1b
fix: style changes
Karlen9 Aug 7, 2024
6e658a1
fix: mobile style fixes
Karlen9 Aug 7, 2024
f697625
feat: first round of apps
Majorfi Aug 7, 2024
ffd01ed
fix: type
Majorfi Aug 7, 2024
604e2f4
fix: menu
Majorfi Aug 7, 2024
2ae47d0
entrance anim
Majorfi Aug 7, 2024
935bc39
feat: styles updates
Karlen9 Aug 7, 2024
7998139
fix: small change
Karlen9 Aug 7, 2024
379d9b1
fix: mobile add card imp
Karlen9 Aug 7, 2024
12cf4dc
feat: add description to apps
Karlen9 Aug 13, 2024
e51f464
feat: add icons
Karlen9 Aug 13, 2024
cc52576
fix: featured apps
Majorfi Aug 27, 2024
97c1c3f
fix: upd images
Majorfi Aug 27, 2024
2686a82
feat: wip sidebar
Karlen9 Aug 27, 2024
584cb1f
feat: changed sizes + hovers
Karlen9 Aug 27, 2024
77f5592
feat: small changes
Karlen9 Aug 28, 2024
98553ad
fix: assets
Majorfi Aug 29, 2024
a66cff9
fix: some spacing
Majorfi Aug 29, 2024
aaeb0e0
fix: design change
Majorfi Aug 29, 2024
c3d7b83
Merge branch 'main' into feat/landing2.0
Majorfi Sep 4, 2024
229ea49
Merge branch 'main' of https://github.com/yearn/yearn.fi into feat/la…
Karlen9 Sep 18, 2024
3c904ae
chore: remove featured app label
Karlen9 Sep 18, 2024
d9267c5
feat: randomize integration apps
Karlen9 Sep 18, 2024
90bf742
feat: moved About to the end of menu
Karlen9 Sep 18, 2024
d8464c1
Merge branch 'feat/landing2.0' of https://github.com/yearn/yearn.fi i…
Karlen9 Sep 18, 2024
560c93f
feat: shuffle all apps
Karlen9 Sep 18, 2024
51e1e3a
Merge branch 'main' of https://github.com/yearn/yearn.fi into feat/la…
Karlen9 Sep 27, 2024
64e19ad
feat: carousel with controls
Karlen9 Sep 27, 2024
c1a4eb1
chore: ts fix
Karlen9 Sep 27, 2024
1613cac
chore: route fix
Karlen9 Sep 30, 2024
91225e9
feat: change some descriptin and catrgory name
Karlen9 Sep 30, 2024
fcc6b09
chore: type fix
Karlen9 Sep 30, 2024
fbf4875
fix: animation with search
Majorfi Oct 1, 2024
5f2ba1a
feat: upd copy
w84april Oct 16, 2024
a238574
feat: redirect
w84april Oct 16, 2024
7be8865
fix: animation bug
w84april Oct 16, 2024
90e55fe
Merge branch 'main' of github.com:yearn/yearn.fi into feat/landing2.0
Karlen9 Nov 19, 2024
76408c6
feat: mobile fixes
Karlen9 Nov 19, 2024
5df1c47
fix: img
w84april Nov 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
44 changes: 44 additions & 0 deletions apps/common/CarouselControls.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import {type ReactElement} from 'react';
import {cl} from '@builtbymom/web3/utils';

type TCarouselControlsProps = {
carouselLength?: number;
onDotsClick: (destination: number) => void;
currentPage: number;
};

export function CarouselControls({
carouselLength = 0,
onDotsClick,
currentPage
}: TCarouselControlsProps): ReactElement | null {
const numberOfControls = Math.ceil(carouselLength / 4);

if (carouselLength && carouselLength < 5) {
return null;
}

return (
<div className={'hidden w-full justify-center md:flex'}>
<div className={'flex gap-x-3'}>
{Array(numberOfControls)
.fill('')
.map((_, index) => (
<button
key={index}
className={'p-[2px]'}
onClick={() => {
onDotsClick(index + 1);
}}>
<div
className={cl(
'size-2 rounded-full',
currentPage === index + 1 ? 'bg-white' : 'bg-gray-500'
)}
/>
</button>
))}
</div>
</div>
);
}
39 changes: 39 additions & 0 deletions apps/common/CarouselSlideArrows.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import {cl} from '@builtbymom/web3/utils';

import {IconChevron} from './icons/IconChevron';

import type {ReactElement} from 'react';

type TCarouselSlideArrowsProps = {
onScrollBack?: VoidFunction;
onScrollForward?: VoidFunction;
className?: string;
};

export function CarouselSlideArrows({
onScrollBack,
onScrollForward,
className
}: TCarouselSlideArrowsProps): ReactElement {
return (
<div className={cl('flex w-full justify-between', className)}>
<div />
<div className={'hidden gap-3 md:flex'}>
<button
onClick={onScrollBack}
className={
'flex !h-8 items-center rounded-[4px] px-4 text-white outline !outline-1 outline-gray-600/50 hover:bg-gray-600/40'
}>
<IconChevron className={'size-3 rotate-90'} />
</button>
<button
onClick={onScrollForward}
className={
'flex !h-8 items-center rounded-[4px] px-4 text-white outline !outline-1 outline-gray-600/50 hover:bg-gray-600/40'
}>
<IconChevron className={'size-3 -rotate-90'} />
</button>
</div>
</div>
);
}
72 changes: 72 additions & 0 deletions apps/common/components/AppCard.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
import Image from 'next/image';
import Link from 'next/link';
import {IconShare} from '@common/icons/IconShare';

import type {ReactElement} from 'react';
import type {TApp} from '@common/types/category';

type TAppCardProps = {
app: TApp;
};

export function AppCard(props: TAppCardProps): ReactElement {
return (
<>
<Link
href={props.app.appURI ?? ''}
target={'_blank'}
className={
'bg-grey-900 group relative hidden h-[240px] min-w-[208px] max-w-[208px] overflow-hidden rounded-lg border border-gray-700/50 p-6 hover:bg-gray-600/40 md:block'
}>
<div className={'mb-4'}>
<div
className={
'absolute right-2 top-2 hidden size-10 items-center justify-center rounded-lg bg-gray-900 group-hover:flex'
}>
<IconShare className={'size-[10px]'} />
</div>
{props.app.logoURI ? (
<Image
src={props.app.logoURI}
alt={props.app.name}
unoptimized
width={240}
height={240}
className={'size-[80px] rounded-full border border-[#292929]/80 object-contain'}
/>
) : (
<div className={'size-[80px] rounded-full bg-fallback'} />
)}
</div>
<div className={'mb-1 text-lg font-bold text-white'}>{props.app.name}</div>

<p className={'max-h-[60px] whitespace-normal text-sm text-gray-400'}>{props.app.description}</p>
</Link>
<Link
href={props.app.appURI}
className={'flex items-center md:hidden'}>
<div>
{props.app.logoURI ? (
<div className={'size-16 rounded-[32px]'}>
<Image
src={props.app.logoURI}
alt={props.app.name}
width={300}
height={300}
unoptimized
className={'size-full rounded-2xl bg-center object-cover md:rounded-[32px]'}
/>
</div>
) : (
<div className={'size-16 rounded-2xl bg-fallback md:rounded-[32px]'} />
)}
</div>

<div className={'ml-4'}>
<div className={'mb-1 text-base font-bold text-gray-300'}>{props.app.name}</div>
<p className={'line-clamp-2 h-12 text-xs text-gray-400 md:text-base'}>{props.app.description}</p>
</div>
</Link>
</>
);
}
82 changes: 82 additions & 0 deletions apps/common/components/AppsCarousel.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import {type ForwardedRef, forwardRef, type ReactElement} from 'react';
import React from 'react';
import {cl} from '@builtbymom/web3/utils';

import {AppCard} from './AppCard';
import {FeaturedApp} from './FeaturedApp';

import type {TApp} from '@common/types/category';

export const AppsCarousel = forwardRef(
(
props: {onScroll?: VoidFunction; isUsingFeatured?: boolean; apps: TApp[]},
ref: ForwardedRef<HTMLDivElement>
): ReactElement => {
return (
<div className={props.isUsingFeatured ? 'h-[262px]' : 'h-[360px] md:h-[262px]'}>
<section className={'absolute left-0 -mx-1 w-full'}>
<div
className={
'pointer-events-none absolute left-0 top-0 z-30 h-[272px] w-1/6 bg-gradient-to-r from-gray-900/0 to-transparent md:h-full'
}
/>
<div
className={
'pointer-events-none absolute right-0 top-0 z-30 h-[272px] w-1/5 bg-gradient-to-l from-gray-900/0 to-transparent md:h-full'
}
/>
<div
ref={ref}
onScroll={props.onScroll}
className={cl(
'hidden md:flex overflow-x-auto pb-1 pl-[38px] scrollbar-none max-sm:pr-6',
props.isUsingFeatured ? 'gap-x-8' : 'flex-col md:flex-row gap-x-4 overflow-y-hidden'
)}>
{props.apps?.map((app, i) => {
return (
<React.Fragment key={app.appURI + i}>
{props.isUsingFeatured ? (
<FeaturedApp
key={app.name + i}
app={app}
/>
) : (
<AppCard
app={app}
key={app.name + i}
/>
)}
</React.Fragment>
);
})}
</div>
<div
ref={ref}
onScroll={props.onScroll}
className={cl(
'flex md:hidden overflow-x-auto pb-1 pl-[38px] scrollbar-none max-sm:pr-6',
props.isUsingFeatured ? 'gap-x-8' : 'flex-col md:flex-row gap-y-4 overflow-y-hidden'
)}>
{props.apps?.slice(0, 4).map((app, i) => {
return (
<React.Fragment key={app.appURI + i}>
{props.isUsingFeatured ? (
<FeaturedApp
key={app.name + i}
app={app}
/>
) : (
<AppCard
app={app}
key={app.name + i}
/>
)}
</React.Fragment>
);
})}
</div>
</section>
</div>
);
}
);
131 changes: 131 additions & 0 deletions apps/common/components/CategorySection.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
import {type ReactElement, useRef, useState} from 'react';
import {useMountEffect} from '@react-hookz/web';
import {CarouselControls} from '@common/CarouselControls';
import {CarouselSlideArrows} from '@common/CarouselSlideArrows';
import {IconShare} from '@common/icons/IconShare';

import {AppsCarousel} from './AppsCarousel';

import type {TApp} from '@common/types/category';

type TAppSectionProps = {
title: string;
onExpandClick: () => void;
apps: TApp[];
};

export const CategorySection = ({title, onExpandClick, apps}: TAppSectionProps): ReactElement => {
const [shuffledApps, set_shuffledApps] = useState<TApp[]>([]);
const [currentPage, set_currentPage] = useState(1);
const carouselRef = useRef<HTMLDivElement | null>(null);
const [isProgrammaticScroll, set_isProgrammaticScroll] = useState(false);

/**********************************************************************************************
** Handles scrolling back to the previous page in the carousel.
** It updates the scroll position, current page, and sets a flag to indicate programmatic
** scrolling. The flag is reset after a delay to allow for smooth scrolling.
*********************************************************************************************/
const onScrollBack = (): void => {
if (!carouselRef.current || currentPage === 1) return;
set_isProgrammaticScroll(true);
carouselRef.current.scrollLeft -= 880;
set_currentPage(prev => prev - 1);

setTimeout(() => {
set_isProgrammaticScroll(false);
}, 3000);
};

/**********************************************************************************************
** Handles scrolling forward to the next page in the carousel.
** It updates the scroll position, current page, and sets a flag to indicate programmatic
** scrolling. The flag is reset after a delay to allow for smooth scrolling.
*********************************************************************************************/
const onScrollForward = (): void => {
if (!carouselRef.current || currentPage === Math.ceil(apps.length / 4)) return;
set_isProgrammaticScroll(true);
carouselRef.current.scrollLeft += 880;
set_currentPage(prev => prev + 1);

setTimeout(() => {
set_isProgrammaticScroll(false);
}, 3000);
};

/**********************************************************************************************
** Handles clicking on the carousel dots to navigate to a specific page.
** It updates the scroll position, current page, and sets a flag to indicate programmatic
** scrolling. The flag is reset after a delay to allow for smooth scrolling.
*********************************************************************************************/
const onDotsClick = (destination: number): void => {
if (!carouselRef.current || destination === currentPage) return;
set_isProgrammaticScroll(true);
if (destination > currentPage) {
carouselRef.current.scrollLeft += 1000 * (destination - currentPage);
setTimeout(() => {
set_isProgrammaticScroll(false);
}, 3000);
} else {
carouselRef.current.scrollLeft -= 1000 * (currentPage - destination);
setTimeout(() => {
set_isProgrammaticScroll(false);
}, 3000);
}
set_currentPage(destination);
};

/**********************************************************************************************
** Handles the scroll event of the carousel.
** It calculates the current page based on the scroll position and updates the state.
** This function is not triggered during programmatic scrolling to avoid conflicts.
*********************************************************************************************/
const onScroll = (): void => {
if (!carouselRef.current || isProgrammaticScroll) return;
const {scrollLeft} = carouselRef.current;
const page = Math.ceil(scrollLeft / 1000) + 1;
set_currentPage(page);
};

/**********************************************************************************************
** On component mount we shuffle the array of Partners to avoid any bias.
**********************************************************************************************/
useMountEffect(() => {
if (apps?.length < 1) {
return;
}
set_shuffledApps(apps?.toSorted(() => 0.5 - Math.random()));
});
return (
<div className={'flex flex-col overflow-hidden'}>
<div className={'mb-6 flex h-10 w-full items-center justify-between pr-1'}>
<div className={'flex gap-x-4'}>
<div className={'whitespace-nowrap text-lg font-bold text-white'}>{title}</div>
<button
onClick={onExpandClick}
className={
'flex items-center rounded-[4px] px-4 py-2 outline !outline-1 outline-gray-600/50 hover:bg-gray-600/40'
}>
<span className={'mr-2 whitespace-nowrap text-xs text-white'}>{'View all'}</span>
<IconShare className={'size-3 text-white'} />
</button>
</div>
{apps?.length > 4 && (
<CarouselSlideArrows
onScrollBack={onScrollBack}
onScrollForward={onScrollForward}
/>
)}
</div>
<AppsCarousel
apps={shuffledApps}
ref={carouselRef}
onScroll={onScroll}
/>
<CarouselControls
carouselLength={apps.length}
onDotsClick={onDotsClick}
currentPage={currentPage}
/>
</div>
);
};
Loading
Loading