Skip to content

Commit f7adff0

Browse files
authored
Merge pull request #85 from kc3hack/RAIT-09/develop
Rait 09/develop
2 parents 27ba6dc + 9bc72a8 commit f7adff0

File tree

9 files changed

+208
-53
lines changed

9 files changed

+208
-53
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
// サーバアクション
2+
'use server';
3+
4+
import { hc } from 'hono/client';
5+
import { AppType } from '../../../../../backend/src/index';
6+
7+
const client = hc<AppType>(process.env.BACKEND_URL ?? 'http://localhost:8080');
8+
9+
/**
10+
* 雅を増やす非同期関数
11+
* @async
12+
* @function addMiyabi
13+
* @param {Object} params - 雅追加のためのパラメータオブジェクト
14+
* @param {string} params.iconUrl - ユーザのアイコン画像URL
15+
* @param {string} params.postId - 雅する投稿のID
16+
*/
17+
export const addMiyabi = async ({ iconUrl, postId }: { iconUrl: string; postId: string }) => {
18+
try {
19+
const res = await client.miyabi.$post({
20+
json: {
21+
post_id: postId,
22+
my_icon: iconUrl,
23+
},
24+
});
25+
26+
// エラーがある場合はログを出力
27+
if (!res.ok) {
28+
console.log(res.statusText);
29+
}
30+
} catch (error) {
31+
console.error(error);
32+
}
33+
};
34+
35+
/**
36+
* 雅を減らす非同期関数
37+
* @async
38+
* @function removeMiyabi
39+
* @param {Object} params - 雅追加のためのパラメータオブジェクト
40+
* @param {string} params.iconUrl - ユーザのアイコン画像URL
41+
* @param {string} params.postId - 雅する投稿のID
42+
*/
43+
export const removeMiyabi = async ({ iconUrl, postId }: { iconUrl: string; postId: string }) => {
44+
try {
45+
const res = await client.miyabi.$delete({
46+
json: {
47+
post_id: postId,
48+
my_icon: iconUrl,
49+
},
50+
});
51+
52+
// エラーがある場合はログを出力
53+
if (!res.ok) {
54+
console.log(res.statusText);
55+
}
56+
} catch (error) {
57+
console.error(error);
58+
}
59+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
// サーバアクション
2+
'use server';
3+
4+
import { hc } from 'hono/client';
5+
import { AppType } from '../../../../../backend/src/index';
6+
7+
const client = hc<AppType>(process.env.BACKEND_URL ?? 'http://localhost:8080');
8+
9+
/**
10+
* 投稿データを削除する非同期関数
11+
* @async
12+
* @function deletePost
13+
* @param {Object} params - 投稿データ取得のためのパラメータオブジェクト
14+
* @param {string} params.iconUrl - ユーザのアイコン画像URL
15+
* @param {string} params.postId - 削除する投稿のID
16+
* @returns {Promise<boolean>} 結果を返すPromise.
17+
*/
18+
const deletePost = async ({
19+
iconUrl,
20+
postId,
21+
}: {
22+
iconUrl: string;
23+
postId: string;
24+
}): Promise<boolean> => {
25+
try {
26+
const res = await client.post.$delete({
27+
json: {
28+
user_icon: iconUrl,
29+
post_id: postId,
30+
},
31+
});
32+
33+
// エラーがある場合はログを出力
34+
if (!res.ok) {
35+
console.log(res.statusText);
36+
return false;
37+
}
38+
return true;
39+
} catch (error) {
40+
console.error(error);
41+
return false;
42+
}
43+
};
44+
45+
export default deletePost;

frontend/src/app/timeline/actions/fetchPosts.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,9 @@ const fetchPosts = async ({
3434
post_id: offsetId,
3535
},
3636
});
37-
console.log(`Loading more Posts... limit: ${limit}, offsetId: ${offsetId}`);
37+
console.log(
38+
`Loading more Posts...\nlimit: ${limit}\niconUrl: ${iconUrl}\noffsetId: ${offsetId}`
39+
);
3840

3941
// エラーがある場合は空の配列を返す
4042
if (!res.ok) {
@@ -44,7 +46,6 @@ const fetchPosts = async ({
4446

4547
const json = await res.json();
4648
return json.posts.map((post) => ({
47-
...post,
4849
id: post.id,
4950
tanka: post.tanka,
5051
original: post.original,

frontend/src/app/timeline/page.tsx

+27-10
Original file line numberDiff line numberDiff line change
@@ -13,13 +13,14 @@ import { useSession } from 'next-auth/react';
1313
import LoginDialog from '@/components/LoginDialog';
1414
import { useRouter } from 'next/navigation';
1515

16-
const LIMIT = 10;
16+
const LIMIT = 10; // 一度に取得する投稿数
17+
const MAX = 100; // タイムラインに表示できる最大投稿数
1718

1819
const Timeline = () => {
1920
// 投稿データの配列
2021
const [posts, setPosts] = useState<PostTypes[]>([]);
2122
// 投稿取得時のオフセットID
22-
const [offsetId, setOffsetId] = useState('');
23+
const offsetIdRef = useRef('');
2324
// データ取得中かどうかのフラグ
2425
const [isLoading, setIsLoading] = useState(false);
2526
// これ以上取得できる投稿があるかのフラグ
@@ -36,6 +37,8 @@ const Timeline = () => {
3637

3738
//IntersectionObserverを保持するためのref
3839
const observer = useRef<IntersectionObserver | null>(null);
40+
// 投稿取得の重複実行を防ぐためのref
41+
const isFetchingRef = useRef(false);
3942

4043
/**
4144
* 追加の投稿データを取得し,状態を更新する非同期関数のCallback Ref
@@ -44,16 +47,24 @@ const Timeline = () => {
4447
* @returns {Promise<void>} 投稿データの取得と状態更新が完了するPromise
4548
*/
4649
const loadMorePosts = useCallback(async () => {
50+
if (isFetchingRef.current) return;
51+
isFetchingRef.current = true;
4752
setIsLoading(true);
4853
// 投稿データを取得
4954
const newPosts = await fetchPosts({
5055
limit: LIMIT,
51-
iconUrl: session.data?.user?.id,
52-
offsetId: offsetId,
56+
iconUrl: session.data?.user?.image ?? '',
57+
offsetId: offsetIdRef.current,
5358
});
5459
if (newPosts && newPosts.length > 0) {
55-
setPosts((prevPosts) => [...prevPosts, ...newPosts]);
56-
setOffsetId(() => newPosts[newPosts.length - 1].id);
60+
setPosts((prevPosts) => {
61+
const updatedPosts = [...prevPosts, ...newPosts];
62+
if (updatedPosts.length >= MAX) {
63+
setHasMore(false);
64+
}
65+
return updatedPosts;
66+
});
67+
offsetIdRef.current = newPosts[newPosts.length - 1].id;
5768
// 取得した投稿数がLIMIT未満の場合は,これ以上取得できる投稿は無い.
5869
if (newPosts.length < LIMIT) {
5970
setHasMore(false);
@@ -63,7 +74,8 @@ const Timeline = () => {
6374
setHasMore(false);
6475
}
6576
setIsLoading(false);
66-
}, [offsetId]);
77+
isFetchingRef.current = false;
78+
}, [session.data?.user?.image]);
6779

6880
// ターゲットの要素を監視するためのcallback ref
6981
const targetRef = useCallback(
@@ -94,6 +106,11 @@ const Timeline = () => {
94106
// eslint-disable-next-line react-hooks/exhaustive-deps
95107
}, []);
96108

109+
// 見かけ上の投稿を削除する関数
110+
const deletePost = (postId: string) => {
111+
setPosts((prevPosts) => prevPosts.filter((post) => post.id !== postId));
112+
};
113+
97114
return (
98115
<div className='relative min-h-screen'>
99116
{/* ヘッダ */}
@@ -117,10 +134,10 @@ const Timeline = () => {
117134
<div className='pt-12'>
118135
<div className='relative mx-auto max-w-lg'>
119136
<SideMenu className='fixed top-16 hidden -translate-x-full lg:block' />
120-
<PostList posts={posts} className='mx-auto max-w-sm lg:max-w-lg' />
137+
<PostList posts={posts} className='mx-auto max-w-sm lg:max-w-lg' onDelete={deletePost} />
121138
{isLoading && <p className='py-3 text-center'>投稿を取得中...</p>}
122139
<div ref={targetRef} className='h-px' />
123-
{!hasMore && <p className='py-3 text-center'>これ以上投稿はありません</p>}
140+
{!hasMore && <p className='py-3 text-center'>これ以上投稿を取得できません。</p>}
124141
</div>
125142
</div>
126143

@@ -146,7 +163,7 @@ const Timeline = () => {
146163
)}
147164

148165
{/* ログイン確認ダイアログ表示が有効の場合,ダイアログを表示する */}
149-
{loginDialogOpen && <LoginDialog isOpen={loginDialogOpen} setIsOpen={setLoginDialogOpen} />}
166+
<LoginDialog isOpen={loginDialogOpen} setIsOpen={setLoginDialogOpen} />
150167
</div>
151168
);
152169
};

frontend/src/components/GifButton.tsx

+4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ export interface GifButtonProps {
99
afterSrc: string;
1010
animationDuration?: number;
1111
initialIsClicked?: boolean;
12+
isAnimationDisabled?: boolean;
1213
onClick?: () => void;
1314
onCancel?: () => void;
1415
className?: string;
@@ -21,6 +22,7 @@ const GifButton = ({
2122
afterSrc,
2223
animationDuration,
2324
initialIsClicked,
25+
isAnimationDisabled = false,
2426
onClick,
2527
onCancel,
2628
className,
@@ -32,6 +34,7 @@ const GifButton = ({
3234
const [isAnimationPlaying, setIsAnimationPlaying] = useState(false);
3335

3436
const clickButton = () => {
37+
if (isAnimationDisabled) return;
3538
setIsAnimationPlaying(true);
3639
setTimeout(() => {
3740
setIsAnimationPlaying(false);
@@ -40,6 +43,7 @@ const GifButton = ({
4043
};
4144

4245
const cancelButton = () => {
46+
if (isAnimationDisabled) return;
4347
setIsAnimationPlaying(false);
4448
setIsClicked(false);
4549
};

frontend/src/components/MiyabiButton.tsx

+3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ interface MiyabiButtonProps {
66
size?: 'small' | 'medium' | 'large';
77
className?: string;
88
initialIsClicked?: boolean;
9+
isAnimationDisabled?: boolean;
910
onClick?: () => void;
1011
onCancel?: () => void;
1112
}
@@ -16,6 +17,7 @@ const MiyabiButton = ({
1617
onCancel,
1718
className,
1819
initialIsClicked = false,
20+
isAnimationDisabled = false,
1921
}: MiyabiButtonProps) => {
2022
return (
2123
<GifButton
@@ -27,6 +29,7 @@ const MiyabiButton = ({
2729
initialIsClicked={initialIsClicked}
2830
onClick={onClick}
2931
onCancel={onCancel}
32+
isAnimationDisabled={isAnimationDisabled}
3033
className={`${className} scale-150 hover:scale-[1.6]`}
3134
/>
3235
);

0 commit comments

Comments
 (0)