Merge branch 'onboarding-suggestions' into 'develop'
Use v2 suggestions endpoint for Onboarding See merge request soapbox-pub/soapbox-fe!1719
This commit is contained in:
commit
819c62fc3c
|
@ -1,49 +1,43 @@
|
||||||
import debounce from 'lodash/debounce';
|
import debounce from 'lodash/debounce';
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { FormattedMessage } from 'react-intl';
|
||||||
import { useDispatch } from 'react-redux';
|
|
||||||
|
|
||||||
import { fetchSuggestions } from 'soapbox/actions/suggestions';
|
|
||||||
import ScrollableList from 'soapbox/components/scrollable_list';
|
import ScrollableList from 'soapbox/components/scrollable_list';
|
||||||
import { Button, Card, CardBody, Stack, Text } from 'soapbox/components/ui';
|
import { Button, Card, CardBody, Stack, Text } from 'soapbox/components/ui';
|
||||||
import AccountContainer from 'soapbox/containers/account_container';
|
import AccountContainer from 'soapbox/containers/account_container';
|
||||||
import { useAppSelector } from 'soapbox/hooks';
|
import useOnboardingSuggestions from 'soapbox/queries/suggestions';
|
||||||
|
|
||||||
const SuggestedAccountsStep = ({ onNext }: { onNext: () => void }) => {
|
const SuggestedAccountsStep = ({ onNext }: { onNext: () => void }) => {
|
||||||
const dispatch = useDispatch();
|
const { data, fetchNextPage, hasNextPage, isFetching } = useOnboardingSuggestions();
|
||||||
|
|
||||||
const suggestions = useAppSelector((state) => state.suggestions.items);
|
|
||||||
const hasMore = useAppSelector((state) => !!state.suggestions.next);
|
|
||||||
const isLoading = useAppSelector((state) => state.suggestions.isLoading);
|
|
||||||
|
|
||||||
const handleLoadMore = debounce(() => {
|
const handleLoadMore = debounce(() => {
|
||||||
if (isLoading) {
|
if (isFetching) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return dispatch(fetchSuggestions());
|
return fetchNextPage();
|
||||||
}, 300);
|
}, 300);
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
dispatch(fetchSuggestions({ limit: 20 }));
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const renderSuggestions = () => {
|
const renderSuggestions = () => {
|
||||||
|
if (!data) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='sm:pt-4 sm:pb-10 flex flex-col'>
|
<div className='sm:pt-4 sm:pb-10 flex flex-col'>
|
||||||
<ScrollableList
|
<ScrollableList
|
||||||
isLoading={isLoading}
|
isLoading={isFetching}
|
||||||
scrollKey='suggestions'
|
scrollKey='suggestions'
|
||||||
onLoadMore={handleLoadMore}
|
onLoadMore={handleLoadMore}
|
||||||
hasMore={hasMore}
|
hasMore={hasNextPage}
|
||||||
useWindowScroll={false}
|
useWindowScroll={false}
|
||||||
style={{ height: 320 }}
|
style={{ height: 320 }}
|
||||||
>
|
>
|
||||||
{suggestions.map((suggestion) => (
|
{data.map((suggestion) => (
|
||||||
<div key={suggestion.account} className='py-2'>
|
<div key={suggestion.account.id} className='py-2'>
|
||||||
<AccountContainer
|
<AccountContainer
|
||||||
// @ts-ignore: TS thinks `id` is passed to <Account>, but it isn't
|
id={suggestion.account.id}
|
||||||
id={suggestion.account}
|
|
||||||
showProfileHoverCard={false}
|
showProfileHoverCard={false}
|
||||||
withLinkToProfile={false}
|
withLinkToProfile={false}
|
||||||
/>
|
/>
|
||||||
|
@ -65,7 +59,7 @@ const SuggestedAccountsStep = ({ onNext }: { onNext: () => void }) => {
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderBody = () => {
|
const renderBody = () => {
|
||||||
if (suggestions.isEmpty()) {
|
if (!data || data.length === 0) {
|
||||||
return renderEmpty();
|
return renderEmpty();
|
||||||
} else {
|
} else {
|
||||||
return renderSuggestions();
|
return renderSuggestions();
|
||||||
|
|
|
@ -0,0 +1,44 @@
|
||||||
|
import { __stub } from 'soapbox/api';
|
||||||
|
import { renderHook, waitFor } from 'soapbox/jest/test-helpers';
|
||||||
|
|
||||||
|
import useOnboardingSuggestions from '../suggestions';
|
||||||
|
|
||||||
|
describe('useCarouselAvatars', () => {
|
||||||
|
describe('with a successul query', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
__stub((mock) => {
|
||||||
|
mock.onGet('/api/v2/suggestions')
|
||||||
|
.reply(200, [
|
||||||
|
{ source: 'staff', account: { id: '1', acct: 'a', account_avatar: 'https://example.com/some.jpg' } },
|
||||||
|
{ source: 'staff', account: { id: '2', acct: 'b', account_avatar: 'https://example.com/some.jpg' } },
|
||||||
|
], {
|
||||||
|
link: '<https://example.com/api/v2/suggestions?since_id=1>; rel=\'prev\'',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('is successful', async() => {
|
||||||
|
const { result } = renderHook(() => useOnboardingSuggestions());
|
||||||
|
|
||||||
|
await waitFor(() => expect(result.current.isFetching).toBe(false));
|
||||||
|
|
||||||
|
expect(result.current.data?.length).toBe(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('with an unsuccessul query', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
__stub((mock) => {
|
||||||
|
mock.onGet('/api/v2/suggestions').networkError();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('is successful', async() => {
|
||||||
|
const { result } = renderHook(() => useOnboardingSuggestions());
|
||||||
|
|
||||||
|
await waitFor(() => expect(result.current.isFetching).toBe(false));
|
||||||
|
|
||||||
|
expect(result.current.error).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,81 @@
|
||||||
|
import { useInfiniteQuery } from '@tanstack/react-query';
|
||||||
|
|
||||||
|
import { fetchRelationships } from 'soapbox/actions/accounts';
|
||||||
|
import { importFetchedAccounts } from 'soapbox/actions/importer';
|
||||||
|
import { getLinks } from 'soapbox/api';
|
||||||
|
import { useApi, useAppDispatch } from 'soapbox/hooks';
|
||||||
|
|
||||||
|
type Account = {
|
||||||
|
acct: string
|
||||||
|
avatar: string
|
||||||
|
avatar_static: string
|
||||||
|
bot: boolean
|
||||||
|
created_at: string
|
||||||
|
discoverable: boolean
|
||||||
|
display_name: string
|
||||||
|
followers_count: number
|
||||||
|
following_count: number
|
||||||
|
group: boolean
|
||||||
|
header: string
|
||||||
|
header_static: string
|
||||||
|
id: string
|
||||||
|
last_status_at: string
|
||||||
|
location: string
|
||||||
|
locked: boolean
|
||||||
|
note: string
|
||||||
|
statuses_count: number
|
||||||
|
url: string
|
||||||
|
username: string
|
||||||
|
verified: boolean
|
||||||
|
website: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type Suggestion = {
|
||||||
|
source: 'staff'
|
||||||
|
account: Account
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export default function useOnboardingSuggestions() {
|
||||||
|
const api = useApi();
|
||||||
|
const dispatch = useAppDispatch();
|
||||||
|
|
||||||
|
const getV2Suggestions = async(pageParam: any): Promise<{ data: Suggestion[], link: string | undefined, hasMore: boolean }> => {
|
||||||
|
const link = pageParam?.link || '/api/v2/suggestions';
|
||||||
|
const response = await api.get<Suggestion[]>(link);
|
||||||
|
const hasMore = !!response.headers.link;
|
||||||
|
const nextLink = getLinks(response).refs.find(link => link.rel === 'next')?.uri;
|
||||||
|
|
||||||
|
const accounts = response.data.map(({ account }) => account);
|
||||||
|
const accountIds = accounts.map((account) => account.id);
|
||||||
|
dispatch(importFetchedAccounts(accounts));
|
||||||
|
dispatch(fetchRelationships(accountIds));
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: response.data,
|
||||||
|
link: nextLink,
|
||||||
|
hasMore,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = useInfiniteQuery(['suggestions', 'v2'], ({ pageParam }) => getV2Suggestions(pageParam), {
|
||||||
|
keepPreviousData: true,
|
||||||
|
getNextPageParam: (config) => {
|
||||||
|
if (config.hasMore) {
|
||||||
|
return { link: config.link };
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = result.data?.pages.reduce<Suggestion[]>(
|
||||||
|
(prev: Suggestion[], curr) => [...prev, ...curr.data],
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
...result,
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
}
|
Loading…
Reference in New Issue